From 425e8cf859b5445318e40259b29d3f5c34fbfb33 Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Thu, 16 Jan 2025 23:29:49 +0200 Subject: [PATCH 01/62] [PR-23724]: added new theme, updated files --- requirements.txt | 2 + source/_static/talkable.css | 697 +----------------- source/_templates/search.html | 65 -- .../converting_into_localization.rst | 4 - source/advanced_features/coupons.rst | 8 +- .../customer_service_portal/overview.rst | 4 - .../customer_service_portal/terminology.rst | 4 - .../customer_service_portal/use_case_1.rst | 4 - .../customer_service_portal/use_case_2.rst | 4 - .../customer_service_portal/use_case_3.rst | 4 - .../customer_service_portal/use_case_4.rst | 4 - .../customer_service_portal/use_case_5.rst | 4 - .../customer_service_portal/use_case_6.rst | 4 - source/advanced_features/file_encryption.rst | 4 - .../advanced_features/passing_custom_data.rst | 4 - .../personal_coupon_sharing.rst | 9 +- source/advanced_features/reg_ex.rst | 4 - source/advanced_features/segments.rst | 5 - source/advanced_features/single_sign_on.rst | 4 - .../subscribing_to_events.rst | 4 - source/advanced_features/url_parameters.rst | 4 - source/advanced_features/utm_tags.rst | 4 - source/android_sdk/advanced.rst | 4 - source/android_sdk/api.rst | 4 - source/android_sdk/deep_linking.rst | 4 - source/android_sdk/getting_started.rst | 4 - .../android_sdk/integration/post_purchase.rst | 4 - source/android_sdk/integration/standalone.rst | 4 - source/android_sdk/testing.rst | 4 - source/android_sdk/upgrade.rst | 4 - source/campaigns.rst | 24 +- source/conf.py | 24 +- source/integration/custom/overview.rst | 26 +- source/ios_sdk/advanced.rst | 4 - source/ios_sdk/api_integration.rst | 4 - source/ios_sdk/example.rst | 4 - source/ios_sdk/integration/post_purchase.rst | 4 - source/ios_sdk/integration/standalone.rst | 4 - .../campaign_placements_description.rst | 65 +- source/web_hooks/check_unsubscribe.rst | 4 - source/web_hooks/claim_signup.rst | 4 - source/web_hooks/create_coupon.rst | 4 - source/web_hooks/event.rst | 5 - source/web_hooks/offer_signup.rst | 4 - source/web_hooks/post_share.rst | 4 - source/web_hooks/referral.rst | 4 - source/web_hooks/reward.rst | 4 - source/web_hooks/unsubscribe.rst | 4 - 48 files changed, 100 insertions(+), 978 deletions(-) delete mode 100644 source/_templates/search.html diff --git a/requirements.txt b/requirements.txt index afee09fd6..ffd6786dc 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,2 +1,4 @@ +furo +sphinx-copybutton Sphinx==8.1.3 sphinx-sitemap==2.6.0 diff --git a/source/_static/talkable.css b/source/_static/talkable.css index 47296eb48..25b0c2481 100644 --- a/source/_static/talkable.css +++ b/source/_static/talkable.css @@ -1,695 +1,34 @@ -@import url("basic.css"); -@import url(https://fonts.googleapis.com/css?family=Roboto:400,500,700,300,400italic); +/* @import url("basic.css"); */ +/* @import url(https://fonts.googleapis.com/css?family=Roboto:400,500,700,300,400italic); */ -/* -- page layout ----------------------------------------------------------- */ -html { - background-color: #EDF0F3; - min-height: 100%; -} - -body { - font: 300 16px/1.45 "Roboto", sans-serif; - color: #3e4f6c; - margin: 0; - padding: 0; -} - -div.admonition, -div.topic, -blockquote, -pre, div[class*="highlight-"] { - clear: none; -} - -div.sidebar::after, -div.topic::after, -div.admonition::after, -blockquote::after, -div.section::after { - clear: none; -} - -.clearfix { - *zoom: 1; -} - -.clearfix:before, -.clearfix:after { - display: table; - content: ""; -} - -.clearfix:after { - clear: both; -} - -.content-container { - margin: auto; - max-width: 960px; - box-sizing: border-box; -} - -a, .a, a:visited { - color: #D36835; - cursor: pointer; - text-decoration: none; -} - -.current > a, -a.current, -a:hover { - color: #3e4f6c; -} - -abbr { - letter-spacing: 0.08em; -} - -.inline { - display: inline-block; -} - -.button { - background: #f0632f; - border: 1px solid #c04f26; - border-radius: 4px; - color: #fff; - cursor: pointer; - display: inline-block; - font: 300 14px/normal "Roboto", sans-serif; - padding: 8px 15px 9px; -} - -.button:hover, -.button:focus { - background-color: #db5523; -} - -.button:active { - background-color: #cb4616; -} - -.textfield { - border: 2px solid #c5cbd4; - border-radius: 4px; - font: 300 14px/normal "Roboto", sans-serif; - height: 36px; - padding: 9px 10px; - width: 100%; - -moz-box-sizing: border-box; - -webkit-box-sizing: border-box; - box-sizing: border-box; - transition: border-color .25s; -} - -.textfield::-moz-placeholder { - color: #ddd; -} - -.textfield::-webkit-input-placeholder { - color: #ddd; -} - -.textfield:focus { - border-color: #90a1b5; -} - -.bodywrapper { - background-color: #fff; - border-radius: 6px; - margin-left: 243px; - margin-bottom: 60px; -} - -.bodywrapper ol, -.bodywrapper ul { - margin-left: 0; - padding-left: 23px; -} - -.bodywrapper blockquote { - margin: 0; -} - -.body { - line-height: 1.5; - padding: 22px 30px 30px; -} - -.body img { - border: 1px solid #ddd; - border-radius: 4px; - box-shadow: 0 5px 15px rgba(0, 0, 0, .2); -} - -.body img.is-minimal { - border: none; - border-radius: 0; - box-shadow: none; - margin: 20px 0; -} - -.nav { - float: left; - line-height: 1.35; - padding: 27px 20px 40px 0; - width: 243px; -} - -.nav ul { - list-style-type: none; - margin: 0; - padding: 0; -} - -.nav li { - margin-bottom: 5px; -} - -.nav ul ul { - font-size: 14px; - margin: 5px 0 5px 20px; -} - -.nav ul ul.simple { - margin-bottom: 0; - margin-top: 0; -} - -/* SEARCH FORM */ -.search-form { - display: inline-block; - vertical-align: middle; - width: 50%; -} - -.search-form-field { - margin-right: 1%; - width: 75%; -} - -/* HEADER */ -.header { - margin: 0 auto; - max-width: 960px; - padding: 20px 0; -} - -.header .button { - width: 23%; -} - -.header-disclaimer { - float: right; - padding-top: 9px; -} - -.header-logo { - background: url("img/logo.svg") no-repeat; - color: transparent; - display: inline-block; - font-size: 0; - height: 32px; - margin-right: 50px; - text-indent: -99999em; - vertical-align: middle; - width: 189px; - background-size: 100%; +.sidebar-brand-text { + display: none; } -blockquote.epigraph { - display: inline-block; - zoom: 1; - margin: 0 -30px 0 -30px; - padding: 30px 70px; - background: #FFE9CE; +.sidebar-logo { + margin: 0; } table.docutils { - width: 100%; + box-shadow: none; + width: 100%; } -table.docutils th, table.docutils td { - border-bottom: 2px solid #e0e0e0; - padding: 8px 8px 8px 0px; - vertical-align: top; -} - -.ptable table { - width: 100%; - margin: 1em 0 1em 0; -} - -.ptable table tr td:first-child, -.ptable table tr th:first-child { - width: 20%; -} - -/* -- body styles ----------------------------------------------------------- */ -p { - margin: 0.8em 0 0.5em 0; -} - -h1, h2, h3, h4, h5, h6 { - font-weight: 200; -} - -h1 { - font-size: 250%; - margin: 0; - padding: 0 0 0.3em 0; - line-height: 1.2em; -} - -h2 { - font-size: 200%; - margin: 1.3em 0 0.2em 0; - padding: 0 0 10px 0; -} - -h3 { - font-size: 160%; - margin: 1em 0 -0.3em 0; - padding-bottom: 5px; -} - -h4 { - font-size: 140%; - margin: 1em 0 -0.3em 0; - padding-bottom: 5px; -} - -div.body h1 a, -div.body h2 a, -div.body h3 a, -div.body h4 a, -div.body h5 a, -div.body h6 a { - color: #657B83 !important; - font-size: 16px; - margin-left: 10px; - vertical-align: middle; -} - -h1 a.anchor, -h2 a.anchor, -h3 a.anchor, -h4 a.anchor, -h5 a.anchor, -h6 a.anchor { - display: none; - margin: 0 0 0 0.3em; - padding: 0 0.2em 0 0.2em; - color: #aaa !important; -} - -h1:hover a.anchor, -h2:hover a.anchor, -h3:hover a.anchor, -h4:hover a.anchor, -h5:hover a.anchor, -h6:hover a.anchor { - display: inline; -} - -h1 a.anchor:hover, -h2 a.anchor:hover, -h3 a.anchor:hover, -h4 a.anchor:hover, -h5 a.anchor:hover, -h6 a.anchor:hover { - color: #777; - background-color: #eee; -} - -cite, code, tt { - font-family: 'Source Code Pro', monospace; - font-size: 0.9em; - letter-spacing: 0.01em; - background-color: #FFF9DA; - font-style: normal; - word-break: break-all; -} - -hr { - border: 1px solid #eee; - margin: 2em 0; -} - -.highlight { - border-left: 4px solid #DADADA; - background: none !important; -} - -pre { - font-family: 'Source Code Pro', monospace; - font-style: normal; - font-size: 14px; - line-height: 1.4; - padding: 15px 20px; - white-space: pre-wrap; /* css-3 */ - white-space: -moz-pre-wrap; /* Mozilla, since 1999 */ - white-space: -pre-wrap; /* Opera 4-6 */ - white-space: -o-pre-wrap; /* Opera 7 */ - word-wrap: break-word; /* Internet Explorer 5.5+ */ -} - -pre a { - color: inherit; - text-decoration: underline; -} - -td.linenos pre { - padding: 0.5em 0; -} - -div.quotebar { - background-color: #f8f8f8; - max-width: 250px; - float: right; - padding: 2px 7px; - border: 1px solid #ccc; -} - -div.topic { - background-color: #f8f8f8; -} - -table { - border-collapse: collapse; - margin: 0; -} - -table td, table th { - padding: 0.2em 0.5em 0.2em 0.5em; -} - -div.admonition { - border-left: 4px solid #D36835; - margin: 2em 0; - padding: 15px 20px; -} - -div.admonition p { - margin: 0.5em 0; -} - -div.admonition pre { - margin: 0.4em 0; -} - -div.admonition p.admonition-title { - color: #D36835; - margin: 0 0 5px; - font-weight: bold; -} - -div.warning, -div.important { - border-left-color: red; -} - -div.warning p.admonition-title, -div.important p.admonition-title { - color: red; -} - -div.hint, -div.tip { - border-left-color: #3e4f6c; -} - -div.hint p.admonition-title, -div.tip p.admonition-title { - color: #3e4f6c; -} - -div.caution, -div.attention, -div.danger, -div.error { - border-left-color: #dc322f; -} - -div.caution p.admonition-title, -div.attention p.admonition-title, -div.danger p.admonition-title, -div.error p.admonition-title { - color: #dc322f; -} - -div.admonition ul, div.admonition ol { - margin: 0.1em 0.5em 0.5em 3em; - padding: 0; -} - -div.versioninfo { - margin: 1em 0 0 0; - border: 1px solid #eee; - background-color: #ddeaf0; - padding: 8px; - line-height: 1.3em; - font-size: 0.9em; -} - -div.viewcode-block:target { - background-color: #f4debf; - border-top: 1px solid #eee; - border-bottom: 1px solid #eee; -} - -ol > li { - margin-top: 15px; -} - -ol > li:first-child { - margin-top: 0; -} - -.reference.internal em { - font-style: normal; -} - -div.hidden { - display: none; -} - -.underline { - text-decoration: underline; -} - -.mtn, .mvn, .man { - margin-top: 0px; -} - -.mrn, .mhn, .man { - margin-right: 0px; -} - -.mbn, .mvn, .man { - margin-bottom: 0px; -} - -.mln, .mhn, .man { - margin-left: 0px; -} - -.mtx, .mvx, .max { - margin-top: 3px; -} - -.mrx, .mhx, .max { - margin-right: 3px; -} - -.mbx, .mvx, .max { - margin-bottom: 3px; -} - -.mlx, .mhx, .max { - margin-left: 3px; -} - -.mts, .mvs, .mas { - margin-top: 5px; -} - -.mrs, .mhs, .mas { - margin-right: 5px; -} - -.mbs, .mvs, .mas { - margin-bottom: 5px; -} - -.mls, .mhs, .mas { - margin-left: 5px; -} - -.mtm, .mvm, .mam { - margin-top: 10px; -} - -.mrm, .mhm, .mam { - margin-right: 10px; -} - -.mbm, .mvm, .mam { - margin-bottom: 10px; -} - -.mlm, .mhm, .mam { - margin-left: 10px; -} - -.mtl, .mvl, .mal { - margin-top: 20px; -} - -.mrl, .mhl, .mal { - margin-right: 20px; -} - -.mbl, .mvl, .mal { - margin-bottom: 20px; -} - -.mll, .mhl, .mal { - margin-left: 20px; -} - -.mtxl, .mvxl, .maxl { - margin-top: 30px; -} - -.mrxl, .mhxl, .maxl { - margin-right: 30px; -} - -.mbxl, .mvxl, .maxl { - margin-bottom: 30px; -} - -.mlxl, .mhxl, .maxl { - margin-left: 30px; -} - -.mtxxl, .mvxxl, .maxxl { - margin-top: 40px; -} - -.mrxxl, .mhxxl, .maxxl { - margin-right: 40px; -} - -.mbxxl, .mvxxl, .maxxl { - margin-bottom: 40px; -} - -.mlxxl, .mhxxl, .maxxl { - margin-left: 40px; -} - -.page-break-after, -.header-printable { - display: none; + vertical-align: top; + padding: 1px 16px 24px 5px; } -/* FIREFOX SPECIFIC */ -@-moz-document url-prefix() { - .button { - padding-bottom: 8px; - padding-top: 6px; - } +body[data-theme="light"] .highlight, +body[data-theme="light"] .highlight button.copybtn, +body[data-theme="light"] .highlight button.copybtn.success:after { + background: #f2f2f2 !important; } -/* MQ */ -@media only screen and (max-width: 980px) { - .header { - padding-left: 20px; - padding-right: 20px; - } - - .header-disclaimer { - font-size: 80%; - } - - .content-container { - padding: 0 20px; - } +.highlight button.copybtn:hover { + stroke: var(--color-link); } -@media only print { - html { - background-color: transparent; - } - - .nav, - .search-form { - display: none !important; - } - - .header { - padding-top: 0; - margin-bottom: 40px; - } - - .header-logo { - background-image: none; - } - - .header-logo:after { - content: url(img/logo.svg); - } - - .section li { - position: relative; - page-break-inside: avoid; - } - - .admonition { - position: relative; - page-break-inside: avoid; - page-break-before: avoid; - } - - .section, - .section li { - widows: 2; - orphans: 2; - } - - .body h1, - .body h2, - .body h3, - .body h4, - .body h5, - .body h6 { - display: block; - position: relative; - page-break-after: avoid; - } - - .body h1 { - text-align: center; - } - - .body img { - display: block; - position: relative; - page-break-before: avoid; - max-height: 700px; - margin: 20px 0; - } - - .body p { - font-size: 20px; - } - - .page-break-after { - display: block; - position: relative; - page-break-after: always; - } +.headerlink { + visibility: hidden !important; } diff --git a/source/_templates/search.html b/source/_templates/search.html deleted file mode 100644 index 431014260..000000000 --- a/source/_templates/search.html +++ /dev/null @@ -1,65 +0,0 @@ -{# - basic/search.html - ~~~~~~~~~~~~~~~~~ - - Template for the search page. - - :copyright: Copyright 2007-2023 by the Sphinx team, see AUTHORS. - :license: BSD, see LICENSE for details. -#} -{%- extends "layout.html" %} -{% set title = _('Search') %} -{%- block scripts %} - {{ super() }} - - -{%- endblock %} -{% block extrahead %} - - {{ super() }} -{% endblock %} -{% block body %} -

{{ _('Search') }}

- {% block scriptwarning %} - - {% endblock %} - {% block searchtext %} -

- {% trans %}Searching for multiple words only shows matches that contain - all words.{% endtrans %} -

- {% endblock %} - {% block searchbox %} -
- - - -
- {% endblock %} - {% block searchresults %} - {% if search_performed %} -

{{ _('Search Results') }}

- {% if not search_results %} -

{{ _('Your search did not match any documents. Please make sure that all words are spelled correctly and that you\'ve selected enough categories.') }}

- {% endif %} - {% endif %} -
- {% if search_results %} - - {% endif %} -
- {% endblock %} -{% endblock %} diff --git a/source/advanced_features/converting_into_localization.rst b/source/advanced_features/converting_into_localization.rst index 25e586161..dabd7e40a 100644 --- a/source/advanced_features/converting_into_localization.rst +++ b/source/advanced_features/converting_into_localization.rst @@ -191,7 +191,3 @@ In order to chieve this AB test we need to All other nested children can be styled following this pattern. 4. Once you’re done with styling it is very easy to set up an AB test, just go back to Campaign Editor and click "Add A/B test variant" link. Once a Campaign goes Live it will start rotating both variants following AB test distribution rules (50:50 by default). - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/coupons.rst b/source/advanced_features/coupons.rst index f16c3f50f..09c959b25 100644 --- a/source/advanced_features/coupons.rst +++ b/source/advanced_features/coupons.rst @@ -45,10 +45,6 @@ Shopify coupon auto-sync Read :ref:`Shopify coupon auto-sync documentation ` for details. -.. container:: hidden - - .. toctree:: - Requirements ------------ @@ -71,6 +67,6 @@ We do not want people confusing 8 with B, 1 with I, 0 with O, L with 1, etc. # Generates a random string from a set of easily readable characters def generate_activation_code(size = 6) - charset = ["2", "3", "4", "6", "7", "9", "A", "C", "D", "E", "F", "G", "H", "J", "K", "M", "N", "P", "Q", "R", "T", "V", "W", "X", "Y", "Z"] - (0...size).map { charset[rand(charset.size)] }.join + charset = ["2", "3", "4", "6", "7", "9", "A", "C", "D", "E", "F", "G", "H", "J", "K", "M", "N", "P", "Q", "R", "T", "V", "W", "X", "Y", "Z"] + (0...size).map { charset[rand(charset.size)] }.join end diff --git a/source/advanced_features/customer_service_portal/overview.rst b/source/advanced_features/customer_service_portal/overview.rst index be446365d..be93ac40b 100644 --- a/source/advanced_features/customer_service_portal/overview.rst +++ b/source/advanced_features/customer_service_portal/overview.rst @@ -30,7 +30,3 @@ To see more personal details, select a user from the list. Example of Customer Service Portal person lookup: .. image:: /_static/img/advanced_features/customer_service_portal/person_lookup.png - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/customer_service_portal/terminology.rst b/source/advanced_features/customer_service_portal/terminology.rst index db831d078..f69be86fb 100644 --- a/source/advanced_features/customer_service_portal/terminology.rst +++ b/source/advanced_features/customer_service_portal/terminology.rst @@ -59,7 +59,3 @@ This is what the person gets as a result of some action (incentive). Available r - **Blocked/No reward**: there was no reward created either because the associated referral was blocked according to Fraud Settings or because of other reasons: incentive criteria has blocked it, or the person was blocklisted. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/customer_service_portal/use_case_1.rst b/source/advanced_features/customer_service_portal/use_case_1.rst index 6bf5173d9..277d1e8b0 100644 --- a/source/advanced_features/customer_service_portal/use_case_1.rst +++ b/source/advanced_features/customer_service_portal/use_case_1.rst @@ -48,7 +48,3 @@ Let’s also expand fraud filters section to check the details: This looks like a valid referral. Let’s move on to :ref:`Use case #2 `. - -.. container:: hidden - - .. toctree:: \ No newline at end of file diff --git a/source/advanced_features/customer_service_portal/use_case_2.rst b/source/advanced_features/customer_service_portal/use_case_2.rst index 0cb4b70f8..218871358 100644 --- a/source/advanced_features/customer_service_portal/use_case_2.rst +++ b/source/advanced_features/customer_service_portal/use_case_2.rst @@ -59,7 +59,3 @@ Here is what email statuses mean: - **Clicked**: Stephanie has clicked on the CTA inside the email. - **Scheduled**: the email will be sent on a specified date. - **Rejected**: the email was rejected by some reason. - -.. container:: hidden - - .. toctree:: \ No newline at end of file diff --git a/source/advanced_features/customer_service_portal/use_case_3.rst b/source/advanced_features/customer_service_portal/use_case_3.rst index 7a0fa6ae7..c6153e03c 100644 --- a/source/advanced_features/customer_service_portal/use_case_3.rst +++ b/source/advanced_features/customer_service_portal/use_case_3.rst @@ -94,7 +94,3 @@ occurs when an Advocate shares with a Friend who purchases and that same Friend in an attempt to get both the Advocate and Friend rewards. We identify this in the same way as described above. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/customer_service_portal/use_case_4.rst b/source/advanced_features/customer_service_portal/use_case_4.rst index 1924dec80..b86418a3d 100644 --- a/source/advanced_features/customer_service_portal/use_case_4.rst +++ b/source/advanced_features/customer_service_portal/use_case_4.rst @@ -51,7 +51,3 @@ flagged/blocked ones. We see that the blocked reason in this case is “Matching “Similar email match”: .. image:: /_static/img/advanced_features/customer_service_portal/marked_as_fraud.jpg - -.. container:: hidden - - .. toctree:: \ No newline at end of file diff --git a/source/advanced_features/customer_service_portal/use_case_5.rst b/source/advanced_features/customer_service_portal/use_case_5.rst index a57d186e8..b21b5d69f 100644 --- a/source/advanced_features/customer_service_portal/use_case_5.rst +++ b/source/advanced_features/customer_service_portal/use_case_5.rst @@ -44,7 +44,3 @@ How do I UnBlocklist a user? If you realize that you made a mistake in Blocklisting a user who should actually be receiving rewards, all you have to do to take a user off the blocklist is to delete their email or IP from the list and then hit “Save Changes” again. It’s that easy. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/customer_service_portal/use_case_6.rst b/source/advanced_features/customer_service_portal/use_case_6.rst index a79bc0599..624f1b558 100644 --- a/source/advanced_features/customer_service_portal/use_case_6.rst +++ b/source/advanced_features/customer_service_portal/use_case_6.rst @@ -50,7 +50,3 @@ After creating the referral manually you will be taken to the newly created refe Advocate reward may be blocked due to incentive criteria, make sure to check the Advocate’s reward status as shown on the screenshot above. - -.. container:: hidden - - .. toctree:: \ No newline at end of file diff --git a/source/advanced_features/file_encryption.rst b/source/advanced_features/file_encryption.rst index 1d47ae910..9a677ddc0 100644 --- a/source/advanced_features/file_encryption.rst +++ b/source/advanced_features/file_encryption.rst @@ -66,7 +66,3 @@ To decrypt a file simply call GPG on it: gpg file_name.asc .. _How To Use GPG to Encrypt and Sign Messages: https://www.digitalocean.com/community/tutorials/how-to-use-gpg-to-encrypt-and-sign-messages#set-up-gpg-keys - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/passing_custom_data.rst b/source/advanced_features/passing_custom_data.rst index 0f3b5db64..699c3c23b 100644 --- a/source/advanced_features/passing_custom_data.rst +++ b/source/advanced_features/passing_custom_data.rst @@ -83,7 +83,3 @@ Key-Value pairs can be referenced calling the desired data key, such as: Any ``custom_properties`` data passed through is tied to the |advocate|, |friend|, or |loyalty_member|. If Talkable receives a custom property that was previously defined for the user, the property gets overwritten with a new value. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/personal_coupon_sharing.rst b/source/advanced_features/personal_coupon_sharing.rst index be0c91a09..4e818b1b4 100644 --- a/source/advanced_features/personal_coupon_sharing.rst +++ b/source/advanced_features/personal_coupon_sharing.rst @@ -7,8 +7,8 @@ Advocate Personal Coupon Sharing ================================ .. note:: - This is an experimental feature, please ask your Customer Success Manager - to apply it into your campaigns. + This is an experimental feature, please ask your Customer Success Manager + to apply it into your campaigns. At Talkable Advocates can invite Friends in two ways: @@ -125,8 +125,3 @@ A: Here is a list of validation rules that are defined globally for any coupons: #. Coupon length is between 3 and 255 characters Any UTF8 characters are allowed (including spaces). - - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/reg_ex.rst b/source/advanced_features/reg_ex.rst index 1e03ae270..5c73de3cb 100644 --- a/source/advanced_features/reg_ex.rst +++ b/source/advanced_features/reg_ex.rst @@ -130,7 +130,3 @@ formats: * `http://site.com/test/products/one` * `http://stage.com/products/two` * `http://test.com/test/products/one/deep?utm=test` - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/segments.rst b/source/advanced_features/segments.rst index bf249f1c7..3df0b41a6 100644 --- a/source/advanced_features/segments.rst +++ b/source/advanced_features/segments.rst @@ -101,8 +101,3 @@ In this example, `segment1`, `segment2`, and `segment3` attributes are passed th Segments can also be passed in `register_affiliate`, `register_purchase`, and `register_event`, providing flexibility for different integration scenarios. This approach simplifies custom data handling for customers, allowing for unified data across various methods and optimizing segmentation management without additional calls. - -.. container:: hidden - - .. toctree:: - diff --git a/source/advanced_features/single_sign_on.rst b/source/advanced_features/single_sign_on.rst index 1d70168e8..cfb325dea 100644 --- a/source/advanced_features/single_sign_on.rst +++ b/source/advanced_features/single_sign_on.rst @@ -92,7 +92,3 @@ Once you've configured SSO, you can test it as followed: 2. Visit the SP-Initiated SSO URL (provided on the SSO configuration page in Talkable) .. image:: /_static/img/advanced_features/sso_sp_initiated_url.png - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/subscribing_to_events.rst b/source/advanced_features/subscribing_to_events.rst index 5bba0a677..3175930c7 100644 --- a/source/advanced_features/subscribing_to_events.rst +++ b/source/advanced_features/subscribing_to_events.rst @@ -51,7 +51,3 @@ Talkable campaigns are equipped with the following set of events: 6. `share_succeeded` — fires each time |advocate| shares. `data.channel` tells which sharing channel was used for the share. 7. `email_gating_passed` — fires when |friend| passes email gating on :ref:`Friend Claim Page `. 8. `email_gating_failed` — fires when |friend| fails to pass email gating on :ref:`Friend Claim Page `. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/url_parameters.rst b/source/advanced_features/url_parameters.rst index bda339fa0..9cfadfa2d 100644 --- a/source/advanced_features/url_parameters.rst +++ b/source/advanced_features/url_parameters.rst @@ -28,7 +28,3 @@ integration. .. note:: Don’t forget to escape URL parameters with URI parameter encoder. - -.. container:: hidden - - .. toctree:: diff --git a/source/advanced_features/utm_tags.rst b/source/advanced_features/utm_tags.rst index 78c008c5f..6bf581a0d 100644 --- a/source/advanced_features/utm_tags.rst +++ b/source/advanced_features/utm_tags.rst @@ -23,7 +23,3 @@ Example of usage: .. code-block:: text https://example1.com/friends?refer=friends&utm_source=example_source&utm_medium=example_medium - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/advanced.rst b/source/android_sdk/advanced.rst index a7f61f3b9..853d9002f 100644 --- a/source/android_sdk/advanced.rst +++ b/source/android_sdk/advanced.rst @@ -192,7 +192,3 @@ your Android application. .. _`handle configuration changes`: https://developer.android.com/guide/topics/resources/runtime-changes.html .. _`Fragment`: https://developer.android.com/guide/fragments .. _`Fragment Transactions`: https://developer.android.com/guide/fragments/transactions - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/api.rst b/source/android_sdk/api.rst index 039e5d2ef..152cdd32d 100644 --- a/source/android_sdk/api.rst +++ b/source/android_sdk/api.rst @@ -230,7 +230,3 @@ Use the ``createSocialShare`` method to track a social share. // Process error } }); - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/deep_linking.rst b/source/android_sdk/deep_linking.rst index 218cc4149..b5e6e26a7 100644 --- a/source/android_sdk/deep_linking.rst +++ b/source/android_sdk/deep_linking.rst @@ -225,7 +225,3 @@ The SDK supports Android 4.1 and later. .. _`GetSocial.im`: https://www.getsocial.im .. _`Android App Links docs`: https://developer.android.com/training/app-links/index.html .. _`Finding SHA256 fingerprint for Android signing keys`: https://docs.getsocial.im/knowledge-base/android-signing-key-sha256/ - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/getting_started.rst b/source/android_sdk/getting_started.rst index 0a44dce7b..eb2ea895c 100644 --- a/source/android_sdk/getting_started.rst +++ b/source/android_sdk/getting_started.rst @@ -162,7 +162,3 @@ Requirements The SDK supports Android 4.1 and later. .. _`Talkable SDK framework`: https://talkable-downloads.s3.amazonaws.com/android-sdk/talkable-sdk.aar - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/integration/post_purchase.rst b/source/android_sdk/integration/post_purchase.rst index 0ca3ef434..d58f58b73 100644 --- a/source/android_sdk/integration/post_purchase.rst +++ b/source/android_sdk/integration/post_purchase.rst @@ -58,7 +58,3 @@ Here is an example of a Purchase capturing, this action should be triggered on t If Post Purchase campaign does not show up when testing make sure you have it Live on the Campaigns listing with `android-post-purchase` tag or the tag you specified in `setCampaignTag`. - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/integration/standalone.rst b/source/android_sdk/integration/standalone.rst index b072b19ca..35c42f115 100644 --- a/source/android_sdk/integration/standalone.rst +++ b/source/android_sdk/integration/standalone.rst @@ -65,7 +65,3 @@ flow and show the :ref:`Advocate Share Form `. } }); ... - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/testing.rst b/source/android_sdk/testing.rst index 1bba47163..9be7f8167 100644 --- a/source/android_sdk/testing.rst +++ b/source/android_sdk/testing.rst @@ -36,7 +36,3 @@ will be applied only if debug is set to `true`. .. warning:: Debug mode must be turned off for release builds. Leaving it enabled will break the referral cycle and expose you to fraud. - -.. container:: hidden - - .. toctree:: diff --git a/source/android_sdk/upgrade.rst b/source/android_sdk/upgrade.rst index 64f4b898a..f0fe7b90c 100644 --- a/source/android_sdk/upgrade.rst +++ b/source/android_sdk/upgrade.rst @@ -283,7 +283,3 @@ Look in :ref:`installation section ` for more detailed info ----- It’s initial release, nothing to do. - -.. container:: hidden - - .. toctree:: diff --git a/source/campaigns.rst b/source/campaigns.rst index 980d8a001..b8f5b5876 100644 --- a/source/campaigns.rst +++ b/source/campaigns.rst @@ -41,12 +41,12 @@ To start editing Views simply visit ``Editor`` page from the Campaign dashboard.

Campaign Structure

-+-------------------------------+-----------------------------+ -| .. toctree:: | | -| :maxdepth: 2 | | -| | | -| campaigns/views | | -+-------------------------------+-----------------------------+ ++-------------------------------+ +| .. toctree:: | +| :maxdepth: 2 | +| | +| campaigns/views | ++-------------------------------+ .. raw:: html @@ -65,9 +65,9 @@ To start editing Views simply visit ``Editor`` page from the Campaign dashboard.

Designer Guide

-+-------------------------------+-----------------------------+ -| .. toctree:: | | -| :maxdepth: 2 | | -| | | -| campaigns/designer | | -+-------------------------------+-----------------------------+ ++-------------------------------+ +| .. toctree:: | +| :maxdepth: 2 | +| | +| campaigns/designer | ++-------------------------------+ diff --git a/source/conf.py b/source/conf.py index 857a33319..9ba32d7fc 100644 --- a/source/conf.py +++ b/source/conf.py @@ -28,7 +28,7 @@ # 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'] +extensions = ['sphinx_sitemap', 'sphinx_copybutton'] sitemap_url_scheme = "{link}" # Add any paths that contain templates here, relative to this directory. @@ -100,20 +100,22 @@ # The theme to use for HTML and HTML Help pages. See the documentation for # a list of builtin themes. -html_theme = 'basic' -html_style = 'talkable.css' +html_theme = "furo" +# html_style = 'talkable.css' # Theme options are theme-specific and customize the look and feel of a theme # further. For a list of options available for each theme, see the # documentation. -#html_theme_options = {} + +html_copy_source = False +html_show_sourcelink = False # Add any paths that contain custom themes here, relative to this directory. #html_theme_path = [] # The name for this set of Sphinx documents. If None, it defaults to # " v documentation". -html_title = u'Talkable Documentation' +html_title = u'' # A shorter title for the navigation bar. Default is the same as html_title. #html_short_title = None @@ -131,6 +133,18 @@ # 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_css_files = [ + 'talkable.css', +] +html_theme_options = { + "light_logo": "img/logo.svg", + "dark_logo": "img/logo.svg", + "light_css_variables": { + "color-brand-primary": "#D36835", + "color-brand-content": "#D36835", + "color-brand-visited": "#D36835", + }, +} # Add any extra paths that contain custom files (such as robots.txt or # .htaccess) here, relative to this directory. These files are copied diff --git a/source/integration/custom/overview.rst b/source/integration/custom/overview.rst index 8ba24a744..24cbd6ef6 100644 --- a/source/integration/custom/overview.rst +++ b/source/integration/custom/overview.rst @@ -10,15 +10,15 @@ Integration High-Level Overview Integrating Talkable referral marketing platform to e-commerce sites is done in a few quick steps by adding the below scripts to your site. - .. note:: +.. note:: - All design and referral campaign setup is done inside the Talkable platform, - then via ‘iframe’ referral campaign content is injected into the e-commerce - site. The below scripts also handle the collection of specific necessary user - data for Talkable to service the referral campaigns. + All design and referral campaign setup is done inside the Talkable platform, + then via ‘iframe’ referral campaign content is injected into the e-commerce + site. The below scripts also handle the collection of specific necessary user + data for Talkable to service the referral campaigns. - Site-specific integration scripts are found after logging in at - ``https://www.admin.talkable.com/sites//integration/other`` + Site-specific integration scripts are found after logging in at + ``https://www.admin.talkable.com/sites//integration/other`` .. raw:: html @@ -35,14 +35,14 @@ a few quick steps by adding the below scripts to your site. Surface purchase detail variables as described in more detail in :ref:`Integration Components `. - .. note:: +.. note:: - If a business has both one-time purchases and subscription transactions, or - the business has a strictly subscription or SaaS model we recommend integrating - the ‘Post Purchase’ through Events. :ref:`Learn more `. + If a business has both one-time purchases and subscription transactions, or + the business has a strictly subscription or SaaS model we recommend integrating + the ‘Post Purchase’ through Events. :ref:`Learn more `. - Post Purchase step is the only step which changes for subscription business - model integration - other steps remain the same. + Post Purchase step is the only step which changes for subscription business + model integration - other steps remain the same. 3. To host the Invite Standalone campaign, create a new HTML page with standard header/footer. Best practice is to create this page on ``www.your-site.com/share`` diff --git a/source/ios_sdk/advanced.rst b/source/ios_sdk/advanced.rst index 5ec35562b..fb2e55349 100644 --- a/source/ios_sdk/advanced.rst +++ b/source/ios_sdk/advanced.rst @@ -148,7 +148,3 @@ See all debugging information in your console which can help you to understand w #import // ... [Talkable manager].debug = YES; - -.. container:: hidden - - .. toctree:: diff --git a/source/ios_sdk/api_integration.rst b/source/ios_sdk/api_integration.rst index a810d8604..73655b57b 100644 --- a/source/ios_sdk/api_integration.rst +++ b/source/ios_sdk/api_integration.rst @@ -193,7 +193,3 @@ to fetch the Offer. [[Talkable manager] retrieveOffer:shortUrlCode withHandler:^(NSDictionary* response, NSError* error) { NSDictionary* offerParams = [response objectForKey:TKBLOfferKey]; }]; - -.. container:: hidden - - .. toctree:: diff --git a/source/ios_sdk/example.rst b/source/ios_sdk/example.rst index 243dec7b8..ff2ddabea 100644 --- a/source/ios_sdk/example.rst +++ b/source/ios_sdk/example.rst @@ -9,7 +9,3 @@ In case you want to take a look at the actual code and see it in action we creat If you have any questions, don’t hesitate to email us at support@talkable.com. Happy hacking! .. _`here`: https://talkable-downloads.s3.amazonaws.com/ios-sdk/talkable_sdk_demo.zip - -.. container:: hidden - - .. toctree:: diff --git a/source/ios_sdk/integration/post_purchase.rst b/source/ios_sdk/integration/post_purchase.rst index 41376cc2c..0e2926c14 100644 --- a/source/ios_sdk/integration/post_purchase.rst +++ b/source/ios_sdk/integration/post_purchase.rst @@ -39,7 +39,3 @@ Here is an example of a Purchase capturing, this action should be triggered on t If Post Purchase campaign does not show up when testing, make sure you have it Live on the Campaigns listing with a default tag (`ios-post-purchase`) or with a tag you specified in the `registerOrigin` call. - -.. container:: hidden - - .. toctree:: diff --git a/source/ios_sdk/integration/standalone.rst b/source/ios_sdk/integration/standalone.rst index 74ccce780..005e0f0b8 100644 --- a/source/ios_sdk/integration/standalone.rst +++ b/source/ios_sdk/integration/standalone.rst @@ -37,7 +37,3 @@ Note that `params` are empty, in this case user will see Signup form on :ref:`Ad } }; [[Talkable manager] registerOrigin:TKBLAffiliateMember params:params]; - -.. container:: hidden - - .. toctree:: diff --git a/source/partials/campaign_placements_description.rst b/source/partials/campaign_placements_description.rst index 83dd9540e..3a79a4b16 100644 --- a/source/partials/campaign_placements_description.rst +++ b/source/partials/campaign_placements_description.rst @@ -6,7 +6,9 @@ The list of all Campaign Placements can be found by going to the All Site Settin -**Managing Placements**. +Managing Placements +------------------- + By default you already have all types of Placements created automatically. Campaigns that you create will attach to one of these Placements based on their types. If you want you can create a new Standalone Placement for your site matching another URL and attach another @@ -16,19 +18,25 @@ Standalone Campaigns there. Placements are strictly tied to their Campaign types. You cannot attach Post Purchase Campaign to Standalone Placement because it expects Campaigns that are going to be embedded not being shown as a popup. -**Priority**. +Priority +-------- + The Placements in the list are ordered by priority. In order to show a Campaign, Talkable matches the URL of a Campaign Placement inclusion (marked as “Shown on”) and a URL that comes with each request that users initiate, for example: `https://[your-site]/share`. Whenever multiple Campaign Placements fit the requested URL Talkable picks the first Placement that sits up top out of the suitable ones. Other Placements that sit below it are ignored in this case. -**Visibility**. +Visibility +---------- + It is possible to indicate on which specific pages the Campaigns attached to the Placement should or should not be shown. It allows you to show the Campaign on all pages of your site except few specified in the "Hidden on" section, for example. Also, you can use :ref:`regular expressions` to select pages you want to not show Talkable campaigns on. -**Custom Property Criteria**. +Custom Property Criteria +------------------------ + A Placement can have a set of criteria defined based on :ref:`custom properties `. If all of the criteria are met, the placement will show a campaign. @@ -41,13 +49,12 @@ The criteria can be defined using one of the following operators: .. note:: Custom Property Criteria require integration version 5.1.4 or higher to work. -|hr| - -.. raw:: html -

Referral Placements

+Referral Placements +~~~~~~~~~~~~~~~~~~~ -

Standalone

+Standalone +^^^^^^^^^^ Campaign will show up on the standalone page on your site, for example: |br| `https://[your-site]/share`. @@ -56,9 +63,8 @@ Campaign will be rendered as an inline block in a corresponding Talkable DIV tag :ref:`Learn more about Standalone Placement ` -.. raw:: html - -

Post Purchase

+Post Purchase +^^^^^^^^^^^^^ Campaign will show up on a successful purchase made by the customer on your site. Usually this is an order confirmation page after checkout. @@ -66,9 +72,8 @@ The campaign will look like a popup with an overlay and a close button. :ref:`Learn more about Post Purchase Placement ` -.. raw:: html - -

Floating Widget

+Floating Widget +^^^^^^^^^^^^^^^ Campaign will show up on every page as a floating button in a corner of the screen so customers are able to access Talkable campaigns from anywhere on your site. @@ -77,9 +82,8 @@ and a close button. :ref:`Learn more about Floating Widget Placement ` -.. raw:: html - -

Claim by Name Widget

+Claim by Name Widget +^^^^^^^^^^^^^^^^^^^^ Campaign will show up as a floating button in a corner of the screen on the cart or checkout page on your site, for example: |br| @@ -89,20 +93,19 @@ Clicking on a Claim by Name widget opens a popup with an overlay and a close but :ref:`Learn more about Claim by Name Placement ` -.. raw:: html - -

Gleam

+Gleam +^^^^^ Campaign will show up on every page as a floating bar that shows your customers their coupon codes after they have been rewarded within any other of your Talkable Campaigns. :ref:`Learn more about Gleam Placement ` -.. raw:: html - -

Loyalty Placements

+Loyalty placements +~~~~~~~~~~~~~~~~~~ -

Loyalty Dashboard

+Loyalty Dashboard +^^^^^^^^^^^^^^^^^ Campaign will show up on the loyalty page on your site, for example: |br| `https://[your-site]/loyalty`. @@ -111,9 +114,8 @@ Campaign will be rendered as an inline block in a corresponding Talkable DIV tag :ref:`Learn more about Loyalty Dashboard Placement ` -.. raw:: html - -

Loyalty Widget

+Loyalty Widget +^^^^^^^^^^^^^^ Campaign will show up on every page as a floating widget that allows your customers to convert their points into coupons. @@ -124,11 +126,8 @@ If a customer is not signed in, the widget prompts them to join the loyalty prog .. note:: When both Loyalty Dashboard and Loyalty Widget are matched on the same URL, only Dashboard will be shown. -|hr| - -.. raw:: html - -

Campaigns Rotating

+Campaigns Rotating +~~~~~~~~~~~~~~~~~~ It is worth mentioning that you can also attach multiple Campaigns into one Placement. In this case Talkable will always show only one Campaign based on a random rotation. diff --git a/source/web_hooks/check_unsubscribe.rst b/source/web_hooks/check_unsubscribe.rst index fdd982b59..142f3193c 100644 --- a/source/web_hooks/check_unsubscribe.rst +++ b/source/web_hooks/check_unsubscribe.rst @@ -50,7 +50,3 @@ View category can be one of the following: * advocate_rewards_confirmation * advocate_rewards_paid * friend_rewards_paid - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/claim_signup.rst b/source/web_hooks/claim_signup.rst index a2c36b2b6..55ad8c28d 100644 --- a/source/web_hooks/claim_signup.rst +++ b/source/web_hooks/claim_signup.rst @@ -37,7 +37,3 @@ Friend Email Gating form example:

Payload parameters provided

The sample payload with parameters for Friend Email Gating Webhook is available here: `Friend Email Gating Webhook Payload `_. - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/create_coupon.rst b/source/web_hooks/create_coupon.rst index f2114591f..ba003c5ca 100644 --- a/source/web_hooks/create_coupon.rst +++ b/source/web_hooks/create_coupon.rst @@ -81,7 +81,3 @@ Create Coupon Webhook can now be set up.

Payload parameters provided by Create Coupon Webhook

The sample payload with parameters for Create Coupon Webhook is available here: `Create Coupon Webhook Payload `_. - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/event.rst b/source/web_hooks/event.rst index 16815c0e5..77f3bf667 100644 --- a/source/web_hooks/event.rst +++ b/source/web_hooks/event.rst @@ -30,8 +30,3 @@ or using the :ref:`Talkable backend API `.

Payload parameters provided

The sample payload with parameters for Event Webhook is available here: `Event Webhook Payload `_. - - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/offer_signup.rst b/source/web_hooks/offer_signup.rst index 5f735b616..011bb396c 100644 --- a/source/web_hooks/offer_signup.rst +++ b/source/web_hooks/offer_signup.rst @@ -37,7 +37,3 @@ Advocate Signup Form example:

Payload parameters provided for Advocate Signup Webhook

The sample payload with parameters for Advocate Signup Webhook is available here: `Advocate Signup Webhook Payload `_. - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/post_share.rst b/source/web_hooks/post_share.rst index a83fb68a7..4dacab6fa 100644 --- a/source/web_hooks/post_share.rst +++ b/source/web_hooks/post_share.rst @@ -43,7 +43,3 @@ The sample payload with parameters for Post Share Webhook is available here: `Po first shown to the Advocate. It could be ``null``. For the most up-to-date information about the Advocate, use ``sharer_info`` property which is updated with the email address the Advocate has entered on the Advocate Signup/Share Page. - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/referral.rst b/source/web_hooks/referral.rst index bb2b4e04f..ccc44caee 100644 --- a/source/web_hooks/referral.rst +++ b/source/web_hooks/referral.rst @@ -62,7 +62,3 @@ The sample payload with parameters for Referral Webhook is available here: `Refe .. include:: /partials/coupon_as_reward.rst .. include:: /partials/incentive_types.rst - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/reward.rst b/source/web_hooks/reward.rst index 24dd1d351..5756470b1 100644 --- a/source/web_hooks/reward.rst +++ b/source/web_hooks/reward.rst @@ -104,7 +104,3 @@ Reward reason can be of 5 following general types. .. include:: /partials/incentive_types.rst .. include:: /partials/coupon_as_reward.rst - -.. container:: hidden - - .. toctree:: diff --git a/source/web_hooks/unsubscribe.rst b/source/web_hooks/unsubscribe.rst index f4b66c7c5..7686cec00 100644 --- a/source/web_hooks/unsubscribe.rst +++ b/source/web_hooks/unsubscribe.rst @@ -29,7 +29,3 @@ Talkable sent email.

Payload parameters provided

The sample payload with parameters for Unsubscribe Webhook is available here: `Unsubscribe Webhook Payload `_. - -.. container:: hidden - - .. toctree:: From 9652fd3bc6ff59570d071da761849590a70bccae Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Fri, 17 Jan 2025 13:13:41 +0200 Subject: [PATCH 02/62] [PR-23724]: updated styles --- source/_static/talkable.css | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/source/_static/talkable.css b/source/_static/talkable.css index 25b0c2481..ca2bb9c1e 100644 --- a/source/_static/talkable.css +++ b/source/_static/talkable.css @@ -32,3 +32,7 @@ body[data-theme="light"] .highlight button.copybtn.success:after { .headerlink { visibility: hidden !important; } + +.hidden { + display: none; +} From 506bae7de681ec120202eaf7d44f98a4ff04e1a3 Mon Sep 17 00:00:00 2001 From: en-ver Date: Wed, 22 Jan 2025 19:46:56 +0300 Subject: [PATCH 03/62] 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 04/62] 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 05/62] 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 191f97db8b17e347377964e796097d21999ed629 Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Thu, 23 Jan 2025 17:08:53 +0200 Subject: [PATCH 06/62] [PR-23724]: moved docs name hide to conf --- source/_static/talkable.css | 7 ------- source/conf.py | 1 + 2 files changed, 1 insertion(+), 7 deletions(-) diff --git a/source/_static/talkable.css b/source/_static/talkable.css index ca2bb9c1e..963b18aa1 100644 --- a/source/_static/talkable.css +++ b/source/_static/talkable.css @@ -1,10 +1,3 @@ -/* @import url("basic.css"); */ -/* @import url(https://fonts.googleapis.com/css?family=Roboto:400,500,700,300,400italic); */ - -.sidebar-brand-text { - display: none; -} - .sidebar-logo { margin: 0; } diff --git a/source/conf.py b/source/conf.py index 9ba32d7fc..827941e0a 100644 --- a/source/conf.py +++ b/source/conf.py @@ -139,6 +139,7 @@ html_theme_options = { "light_logo": "img/logo.svg", "dark_logo": "img/logo.svg", + "sidebar_hide_name": True, "light_css_variables": { "color-brand-primary": "#D36835", "color-brand-content": "#D36835", From c5dafe36cedc54d3a560b09689f15ff6ca1c0777 Mon Sep 17 00:00:00 2001 From: en-ver Date: Fri, 24 Jan 2025 10:39:56 +0300 Subject: [PATCH 07/62] 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 8baf0080488eeb90f9c261a955d761c97e8da8a6 Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Fri, 24 Jan 2025 19:21:09 +0200 Subject: [PATCH 08/62] [PR-23724]: updated campaign placements image tag --- source/partials/campaign_placements_description.rst | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/source/partials/campaign_placements_description.rst b/source/partials/campaign_placements_description.rst index 3a79a4b16..54a902756 100644 --- a/source/partials/campaign_placements_description.rst +++ b/source/partials/campaign_placements_description.rst @@ -2,9 +2,7 @@ Campaign Placements are the places (pages) on your site where the Campaigns are The list of all Campaign Placements can be found by going to the All Site Settings page from the header Menu (top right corner) and choosing the Placements section: -.. raw:: html - - +.. image:: /_static/img/campaign_placements/path.png Managing Placements ------------------- From a373777c2bfec0dec1534289cac679472eb3f4ac Mon Sep 17 00:00:00 2001 From: en-ver Date: Fri, 24 Jan 2025 21:03:37 +0300 Subject: [PATCH 09/62] 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 10/62] 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 11/62] 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 12/62] 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 97b67e13ef15c144854f874535b1dc1777da8747 Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Sun, 26 Jan 2025 22:47:23 +0200 Subject: [PATCH 13/62] [PR-23724]: fixed color for auto theme mode --- source/_static/talkable.css | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/source/_static/talkable.css b/source/_static/talkable.css index 963b18aa1..887465952 100644 --- a/source/_static/talkable.css +++ b/source/_static/talkable.css @@ -18,6 +18,14 @@ body[data-theme="light"] .highlight button.copybtn.success:after { background: #f2f2f2 !important; } +@media (prefers-color-scheme: light) { + body[data-theme="auto"] .highlight, + body[data-theme="auto"] .highlight button.copybtn, + body[data-theme="auto"] .highlight button.copybtn.success:after { + background: #f2f2f2 !important; + } +} + .highlight button.copybtn:hover { stroke: var(--color-link); } From 2a3f5724ee402878a9ecedf2a42c55b4a032053b Mon Sep 17 00:00:00 2001 From: Enver Date: Mon, 27 Jan 2025 20:41:35 +0300 Subject: [PATCH 14/62] [PR-23781] Migrate docs to Jenkins (#358) Co-authored-by: Bohdan Zhuravel --- .circleci/config.yml | 16 +- .env.template | 10 + .github/PULL_REQUEST_TEMPLATE.md | 4 - .gitignore | 20 +- .ruby-gemset | 1 - .ruby-version | 1 - CNAME | 1 - Dockerfile | 14 ++ Gemfile | 11 - Guardfile | 14 -- Procfile | 2 - README-devops.md | 93 ++++++++ README-maintainer.md | 215 ++++++++++++++++++ README.md | 192 +++++++--------- Rakefile | 109 --------- docker-compose.yml | 32 +++ nginx/conf.d/default.conf | 69 ++++++ robots.txt => nginx/robots/development.txt | 3 +- .../robots.txt => nginx/robots/production.txt | 1 + nginx/robots/staging.txt | 4 + packages.txt | 9 + requirements.txt | 38 +++- source/_utils/__init__.py | 1 + source/_utils/baseurl.py | 15 ++ source/conf.py | 107 +++++---- 25 files changed, 666 insertions(+), 316 deletions(-) create mode 100644 .env.template delete mode 100644 .ruby-gemset delete mode 100644 .ruby-version delete mode 100644 CNAME create mode 100644 Dockerfile delete mode 100644 Gemfile delete mode 100644 Guardfile delete mode 100644 Procfile create mode 100644 README-devops.md create mode 100644 README-maintainer.md delete mode 100644 Rakefile create mode 100644 docker-compose.yml create mode 100644 nginx/conf.d/default.conf rename robots.txt => nginx/robots/development.txt (53%) rename source/robots.txt => nginx/robots/production.txt (87%) create mode 100644 nginx/robots/staging.txt create mode 100644 packages.txt create mode 100644 source/_utils/__init__.py create mode 100644 source/_utils/baseurl.py 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 [![](http://proxies.talkable.com/talkable/PR-1234)](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? -[![Build Status](https://circleci.com/gh/talkable/talkable-docs.svg?style=svg&circle-token=cc33458158e7b0c1f6f8cbf1bcbf74f00ee28a8e)](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 -

Hello world!

- ``` - -#### Empty Line at the End of Files - -Always ensure there is a blank line at the end of each file to adhere to formatting standards. - -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 +## Troubleshooting -If you did everything right, deploying is as easy as `rake deploy` from "master" branch. +#### Can't view the documentation locally in the browser? -#### Deploying to Staging +1. Ensure you are using the correct port number and protocol. + The port number should match the value provided in `.env` as `LOCAL_PORT`. -If it’s your first time deploying to staging, run `rake setup` to setup git remote. +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``` -4. Deploy: - ```rake deploy:staging``` + ```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](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/) 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/docker-compose.yml b/docker-compose.yml new file mode 100644 index 000000000..0795ce5c1 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,32 @@ +services: + + sphinx: + build: + context: . + dockerfile: Dockerfile + container_name: sphinx + volumes: + - ./source:/docs/source + - docs_build:/docs/_build + environment: + - ENVIRONMENT=${ENVIRONMENT} + - LOCAL_PORT=${LOCAL_PORT} + - PRODUCTION_HOST=${PRODUCTION_HOST} + - STAGING_HOST=${STAGING_HOST} + - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} + + nginx: + image: nginx:1.27-alpine3.20 + container_name: nginx + volumes: + - docs_build:/var/www/html + - ./nginx/conf.d:/etc/nginx/conf.d + - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt + ports: + - ${LOCAL_PORT}:80 + depends_on: + - sphinx + +volumes: + docs_build: + driver: local diff --git a/nginx/conf.d/default.conf b/nginx/conf.d/default.conf new file mode 100644 index 000000000..5eb1145ca --- /dev/null +++ b/nginx/conf.d/default.conf @@ -0,0 +1,69 @@ +server { + listen 80; + server_name _; + + root /var/www/html; + index index.html; + + # 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; + + # Rewrite .html → no .html + location ~* ^/.+\.html$ { + # Avoid redirect loops if this is an internal sub-request (fallback) + if ($uri != $request_uri) { + break; + } + + # If it ends with "index.html", remove "index.html" + # e.g. /path/index.html => /path/ + # e.g. /index.html => / + rewrite "^/(.*)index\.html$" /$1 permanent; + + # 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; + access_log off; + log_not_found off; + } + + # Default static-file behavior + location / { + try_files $uri $uri/ =404; + } +} diff --git a/robots.txt b/nginx/robots/development.txt similarity index 53% rename from robots.txt rename to nginx/robots/development.txt index 60ed3d057..1f53798bb 100644 --- a/robots.txt +++ b/nginx/robots/development.txt @@ -1,3 +1,2 @@ User-agent: * -Disallow: - +Disallow: / diff --git a/source/robots.txt b/nginx/robots/production.txt similarity index 87% rename from source/robots.txt rename to nginx/robots/production.txt index f3fc00670..4acb8d432 100644 --- a/source/robots.txt +++ b/nginx/robots/production.txt @@ -1,3 +1,4 @@ User-agent: * +Allow: / Sitemap: https://docs.talkable.com/sitemap.xml diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt new file mode 100644 index 000000000..ed8d11f01 --- /dev/null +++ b/nginx/robots/staging.txt @@ -0,0 +1,4 @@ +User-agent: * +Disallow: / + +Sitemap: https://staging-docs.talkable.com/sitemap.xml diff --git a/packages.txt b/packages.txt new file mode 100644 index 000000000..cca18c7b5 --- /dev/null +++ b/packages.txt @@ -0,0 +1,9 @@ +# This is a list of packages used in the framework +# 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 diff --git a/requirements.txt b/requirements.txt index ffd6786dc..7b2f8fbf9 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,4 +1,38 @@ -furo -sphinx-copybutton +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 +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 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 827941e0a..296acbe88 100644 --- a/source/conf.py +++ b/source/conf.py @@ -12,88 +12,96 @@ # All configuration values have a default; values that are commented out # serve to show the default. -# import sys -# import os +import sys +import os # 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 ------------------------------------------------ # 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 +119,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", @@ -150,7 +164,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. @@ -159,26 +173,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 @@ -189,26 +203,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 = baseurl # -- 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 6893b6a0e192990a33c106d08196b31269845a4b Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 22:16:11 +0300 Subject: [PATCH 15/62] rename staging-docs.talkable.com to docs.bastion.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 7d3d25946..4bfe1792a 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=staging-docs.talkable.com +STAGING_HOST=docs.bastion.talkable.com PRODUCTION_HOST=docs.talkable.com diff --git a/README.md b/README.md index ed2a8ba2c..8f88f8b56 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 [staging-docs.talkable.com](https://staging-docs.talkable.com). +- [staging](https://github.com/talkable/talkable-docs/tree/staging): A staging branch used for testing by QA. It is available at [docs.bastion.talkable.com](https://docs.bastion.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 => [staging-docs.talkable.com](https://staging-docs.talkable.com/). +- Commit to the [staging](https://github.com/talkable/talkable-docs/tree/staging) branch => [docs.bastion.talkable.com](https://docs.bastion.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: [staging-docs.talkable.com](https://staging-docs.talkable.com/) +- Staging web server: [docs.bastion.talkable.com](https://docs.bastion.talkable.com/) - GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) - Production web server: [docs.talkable.com](https://docs.talkable.com/) diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt index ed8d11f01..45813aeac 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 +Sitemap: https://docs.bastion.talkable.com/sitemap.xml From 19a0a87d8ac5d11dcca84a216b99356c1873d88f Mon Sep 17 00:00:00 2001 From: Enver Date: Mon, 27 Jan 2025 22:55:12 +0300 Subject: [PATCH 16/62] rename staging-docs.talkable.com to docs.bastion.talkable.com (#360) --- .env.template | 2 +- README.md | 24 ++++-------------------- nginx/robots/staging.txt | 2 +- 3 files changed, 6 insertions(+), 22 deletions(-) diff --git a/.env.template b/.env.template index 7d3d25946..4bfe1792a 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=staging-docs.talkable.com +STAGING_HOST=docs.bastion.talkable.com PRODUCTION_HOST=docs.talkable.com diff --git a/README.md b/README.md index 505e69e69..f89a948fb 100644 --- a/README.md +++ b/README.md @@ -13,10 +13,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). - <<<<<<< HEAD -- # [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 [staging-docs.talkable.com](https://staging-docs.talkable.com). - > > > > > > > vo-PR-23724 +- [staging](https://github.com/talkable/talkable-docs/tree/staging): A staging branch used for testing by QA. It is available at [docs.bastion.talkable.com](https://docs.bastion.talkable.com). - Feature branches created from `master` by individual contributors/developers. ## What is the documentation update workflow? @@ -38,12 +35,7 @@ The repository consists of the following branches: 1. **Navigate to the repository root directory.** -<<<<<<< HEAD -Ensure the `docker-compose.yaml` file is located there. -======= -Ensure the `docker-compose.yml` file is located there. - -> > > > > > > vo-PR-23724 + Ensure the `docker-compose.yml` file is located there. 2. **Create an `.env` file by copying `.env.template`.** @@ -56,11 +48,7 @@ Ensure the `docker-compose.yml` file is located there. Run the command: ```bash - <<<<<<< HEAD - docker-compose up -d - ======= docker compose up -d - >>>>>>> vo-PR-23724 ``` 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. @@ -74,11 +62,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: -<<<<<<< HEAD - -- # 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 => [staging-docs.talkable.com](https://staging-docs.talkable.com/). - > > > > > > > vo-PR-23724 +- Commit to the [staging](https://github.com/talkable/talkable-docs/tree/staging) branch => [docs.bastion.talkable.com](https://docs.bastion.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? @@ -149,6 +133,6 @@ name "Talkable Section". ## Links - GitHub "staging" branch: [staging](https://github.com/talkable/talkable-docs/tree/staging) -- Staging web server: [staging-docs.talkable.com](https://staging-docs.talkable.com/) +- Staging web server: [docs.bastion.talkable.com](https://docs.bastion.talkable.com/) - GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) - Production web server: [docs.talkable.com](https://docs.talkable.com/) diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt index ed8d11f01..45813aeac 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 +Sitemap: https://docs.bastion.talkable.com/sitemap.xml From 4095278df515ef527cf993bbe35bfbdfc284092f Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Mon, 27 Jan 2025 23:57:47 +0200 Subject: [PATCH 17/62] deploy on jenkins --- .gitignore | 90 ++++++++++++++++++++++++++++++++++++++ deploy/Jenkinsfile | 105 +++++++++++++++++++++++++++++++++++++++++++++ deploy/deploy.sh | 0 3 files changed, 195 insertions(+) create mode 100644 deploy/Jenkinsfile create mode 100755 deploy/deploy.sh diff --git a/.gitignore b/.gitignore index c3f20a68f..65a67ae36 100644 --- a/.gitignore +++ b/.gitignore @@ -17,3 +17,93 @@ __pycache__ /build /.bundle Gemfile.lock + +### Intellij+all ### +# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider +# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839 + +# User-specific stuff +.idea/**/workspace.xml +.idea/**/tasks.xml +.idea/**/usage.statistics.xml +.idea/**/dictionaries +.idea/**/shelf + +# AWS User-specific +.idea/**/aws.xml + +# Generated files +.idea/**/contentModel.xml + +# Sensitive or high-churn files +.idea/**/dataSources/ +.idea/**/dataSources.ids +.idea/**/dataSources.local.xml +.idea/**/sqlDataSources.xml +.idea/**/dynamic.xml +.idea/**/uiDesigner.xml +.idea/**/dbnavigator.xml + +# Gradle +.idea/**/gradle.xml +.idea/**/libraries + +# Gradle and Maven with auto-import +# When using Gradle or Maven with auto-import, you should exclude module files, +# since they will be recreated, and may cause churn. Uncomment if using +# auto-import. +# .idea/artifacts +# .idea/compiler.xml +# .idea/jarRepositories.xml +# .idea/modules.xml +# .idea/*.iml +# .idea/modules +# *.iml +# *.ipr + +# CMake +cmake-build-*/ + +# Mongo Explorer plugin +.idea/**/mongoSettings.xml + +# File-based project format +*.iws + +# IntelliJ +out/ + +# mpeltonen/sbt-idea plugin +.idea_modules/ + +# JIRA plugin +atlassian-ide-plugin.xml + +# Cursive Clojure plugin +.idea/replstate.xml + +# SonarLint plugin +.idea/sonarlint/ + +# Crashlytics plugin (for Android Studio and IntelliJ) +com_crashlytics_export_strings.xml +crashlytics.properties +crashlytics-build.properties +fabric.properties + +# Editor-based Rest Client +.idea/httpRequests + +# Android studio 3.1+ serialized cache file +.idea/caches/build_file_checksums.ser + +### Intellij+all Patch ### +# Ignore everything but code style settings and run configurations +# that are supposed to be shared within teams. + +.idea/* + +!.idea/codeStyles +!.idea/runConfigurations + +# End of https://www.toptal.com/developers/gitignore/api/intellij+all diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile new file mode 100644 index 000000000..e8d930eed --- /dev/null +++ b/deploy/Jenkinsfile @@ -0,0 +1,105 @@ +#!/usr/bin/env groovy + +// Upgrades or installs nginx containers for WordPress. +def dockerContainerUpdate(String hostEnv) { + sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-helm/Docker/wp-nginx && \ + git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ + docker-compose ls && docker ps && \ + docker-compose down && docker-compose up -d --remove-orphans && \ + docker-compose ls && docker ps' """ +} + +pipeline { + agent { + label "staging" + } + environment { + GITHUB_ACC = credentials('talkable-bot-deploy-token') + JENKINS_BUILD_USER_ID = "" +// CHANGESET_REGEXP = "^nginx/.*" + } + triggers { + pollSCM('* * * * *') // Polls SCM every minute + } + options { + buildDiscarder(logRotator(numToKeepStr: '15', daysToKeepStr: '14')) + disableConcurrentBuilds() + parallelsAlwaysFailFast() + ansiColor('xterm') + } + stages { + stage('Preparation') { + steps { + script { + wrap([$class: 'BuildUser']) { + JENKINS_BUILD_USER_ID = "${BUILD_USER_ID}" + currentBuild.displayName = "${JOB_BASE_NAME}-${BUILD_NUMBER}" + currentBuild.description = "Build by ${JENKINS_BUILD_USER_ID}" + } + if (env.BRANCH_NAME == 'master') { + env.HOST_ENV = '10.200.1.200' + env.JENKINS_CRED_ID = 'ssh-jenkins-to-prod1-wp-nginx' + } else if (env.BRANCH_NAME == 'staging') { + env.HOST_ENV = '10.1.2.12' + env.JENKINS_CRED_ID = 'ssh-jenkins-to-bastion-wp-nginx' + } else { + echo "Hello from ${BRANCH_NAME} branch! We only build the following branches of the Talkable repository: master, staging." + error("Non-supported branch for build") + } + } + } + } + stage('Build') { + when { + beforeAgent true + changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" + } + steps { + script { + withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { + dockerContainerUpdate('$HOST_ENV') + } + } + } + } + stage('Manual-build') { + when { + not + { + triggeredBy 'SCMTrigger' + } + } + steps { + script { + withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { + dockerContainerUpdate('$HOST_ENV') + } + } + } + } + } + post { + success { + echo 'Build was a success' + } + failure { + echo 'Build failed' + } + unstable { + echo 'Build is unstable' + } + aborted { + echo 'Build was aborted' + } + cleanup { + echo 'Performing cleanup' + cleanWs cleanWhenFailure: false, + cleanWhenUnstable: true, + cleanWhenAborted: true, + cleanWhenNotBuilt: true, + cleanWhenSuccess: true, + deleteDirs: true, + notFailBuild: true + } + } +} diff --git a/deploy/deploy.sh b/deploy/deploy.sh new file mode 100755 index 000000000..e69de29bb From a35a542377b44236a18f952fd04798d2383558b7 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:02:05 +0200 Subject: [PATCH 18/62] deploy on jenkins-2 --- deploy/Jenkinsfile | 1 - 1 file changed, 1 deletion(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index e8d930eed..a43a720e4 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -16,7 +16,6 @@ pipeline { environment { GITHUB_ACC = credentials('talkable-bot-deploy-token') JENKINS_BUILD_USER_ID = "" -// CHANGESET_REGEXP = "^nginx/.*" } triggers { pollSCM('* * * * *') // Polls SCM every minute From 962343692ca82acf7de6acc8105ff8f8c6e79653 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:04:15 +0200 Subject: [PATCH 19/62] deploy on jenkins-3 --- deploy/Jenkinsfile | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index a43a720e4..13d279f2a 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -16,6 +16,7 @@ pipeline { environment { GITHUB_ACC = credentials('talkable-bot-deploy-token') JENKINS_BUILD_USER_ID = "" +// CHANGESET_REGEXP = "^nginx/.*" } triggers { pollSCM('* * * * *') // Polls SCM every minute @@ -49,10 +50,10 @@ pipeline { } } stage('Build') { - when { - beforeAgent true - changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" - } +// when { +// beforeAgent true +// changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" +// } steps { script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { From cb35118cbd5cf4c39cb5bbb021992d754121f378 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:05:37 +0200 Subject: [PATCH 20/62] deploy on jenkins-4 --- deploy/Jenkinsfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 13d279f2a..de078ada8 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,7 +2,7 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { - sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-helm/Docker/wp-nginx && \ + sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ From 4db2f04eaa145fbb96cdeaf2e5e3d33597c37a51 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:16:20 +0200 Subject: [PATCH 21/62] deploy on jenkins-5 --- deploy/Jenkinsfile | 5 +++++ docker-compose.yml | 1 + 2 files changed, 6 insertions(+) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index de078ada8..650657d6c 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -16,6 +16,7 @@ pipeline { environment { GITHUB_ACC = credentials('talkable-bot-deploy-token') JENKINS_BUILD_USER_ID = "" + LOCAL_PORT="8080" // CHANGESET_REGEXP = "^nginx/.*" } triggers { @@ -39,9 +40,13 @@ pipeline { if (env.BRANCH_NAME == 'master') { env.HOST_ENV = '10.200.1.200' env.JENKINS_CRED_ID = 'ssh-jenkins-to-prod1-wp-nginx' + env.ENVIRONMENT="production" + env.PRODUCTION_HOST="docs.talkable.com" } else if (env.BRANCH_NAME == 'staging') { env.HOST_ENV = '10.1.2.12' env.JENKINS_CRED_ID = 'ssh-jenkins-to-bastion-wp-nginx' + env.ENVIRONMENT="staging" + env.STAGING_HOST="docs.bastion.talkable.com" } else { echo "Hello from ${BRANCH_NAME} branch! We only build the following branches of the Talkable repository: master, staging." error("Non-supported branch for build") diff --git a/docker-compose.yml b/docker-compose.yml index 4a23ddfe9..90e107e3c 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,4 +1,5 @@ services: + sphinx: build: context: . From 272b1395650b193100209a721317482703ae7200 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:19:35 +0200 Subject: [PATCH 22/62] deploy on jenkins-6 --- deploy/Jenkinsfile | 2 ++ 1 file changed, 2 insertions(+) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 650657d6c..ae1bd7784 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -61,6 +61,7 @@ pipeline { // } steps { script { + sh 'printenv | sort' withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') } @@ -75,6 +76,7 @@ pipeline { } } steps { + sh 'printenv | sort' script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') From 30af536590184754776060ab4d972038f1d459ab Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:34:08 +0200 Subject: [PATCH 23/62] deploy on jenkins-7 --- deploy/Jenkinsfile | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index ae1bd7784..7a986545f 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,8 +2,15 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { + def envFile = """ +ENVIRONMENT=${env.ENVIRONMENT} +STAGING_HOST=${env.STAGING_HOST ?: ''} +PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} +LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} +""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ + echo "${envFile.trim()}" > .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ From 15ef8f9d004a00e6885009be9402b35ebb537713 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:37:27 +0200 Subject: [PATCH 24/62] deploy on jenkins-8 --- deploy/Jenkinsfile | 2 ++ 1 file changed, 2 insertions(+) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 7a986545f..08dc6cf76 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -6,11 +6,13 @@ def dockerContainerUpdate(String hostEnv) { ENVIRONMENT=${env.ENVIRONMENT} STAGING_HOST=${env.STAGING_HOST ?: ''} PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} +DEVELOPMENT_HOST=${env.DEVELOPMENT_HOST ?: 'localhost'} LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} """ sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ echo "${envFile.trim()}" > .env && \ + cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ From f58976ec5a90b4df6e8a5a218802958bf09d55c1 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:40:37 +0200 Subject: [PATCH 25/62] deploy on jenkins-9 --- deploy/Jenkinsfile | 1 + 1 file changed, 1 insertion(+) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 08dc6cf76..87771ed37 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -56,6 +56,7 @@ pipeline { env.JENKINS_CRED_ID = 'ssh-jenkins-to-bastion-wp-nginx' env.ENVIRONMENT="staging" env.STAGING_HOST="docs.bastion.talkable.com" + env.PRODUCTION_HOST="docs.talkable.com" } else { echo "Hello from ${BRANCH_NAME} branch! We only build the following branches of the Talkable repository: master, staging." error("Non-supported branch for build") From c06d27710a9753ee1733e7c921e69fb30365ce1f Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:46:03 +0200 Subject: [PATCH 26/62] deploy on jenkins-10 --- deploy/Jenkinsfile | 2 -- docker-compose.yml | 11 ++++++++--- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 87771ed37..876fd0746 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -6,7 +6,6 @@ def dockerContainerUpdate(String hostEnv) { ENVIRONMENT=${env.ENVIRONMENT} STAGING_HOST=${env.STAGING_HOST ?: ''} PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} -DEVELOPMENT_HOST=${env.DEVELOPMENT_HOST ?: 'localhost'} LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} """ sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ @@ -56,7 +55,6 @@ pipeline { env.JENKINS_CRED_ID = 'ssh-jenkins-to-bastion-wp-nginx' env.ENVIRONMENT="staging" env.STAGING_HOST="docs.bastion.talkable.com" - env.PRODUCTION_HOST="docs.talkable.com" } else { echo "Hello from ${BRANCH_NAME} branch! We only build the following branches of the Talkable repository: master, staging." error("Non-supported branch for build") diff --git a/docker-compose.yml b/docker-compose.yml index 90e107e3c..33581a98d 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -9,11 +9,16 @@ services: - ./source:/docs/source - docs_build:/docs/_build environment: - - ENVIRONMENT=${ENVIRONMENT} - - LOCAL_PORT=${LOCAL_PORT} + - ENVIRONMENT=staging + - LOCAL_PORT=8080 - PRODUCTION_HOST=${PRODUCTION_HOST} - - STAGING_HOST=${STAGING_HOST} + - STAGING_HOST=docs.bastion.talkable.com - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} +# - ENVIRONMENT=${ENVIRONMENT} +# - LOCAL_PORT=${LOCAL_PORT} +# - PRODUCTION_HOST=${PRODUCTION_HOST} +# - STAGING_HOST=${STAGING_HOST} +# - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} nginx: image: nginx:1.27-alpine3.20 From f95ee9ef677ef6446cb344c45d7667ab9a16d8cd Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 00:52:03 +0200 Subject: [PATCH 27/62] deploy on jenkins-11 --- docker-compose.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/docker-compose.yml b/docker-compose.yml index 33581a98d..53d6b4bd8 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -4,7 +4,7 @@ services: build: context: . dockerfile: Dockerfile - container_name: sphinx + container_name: docs-sphinx-${ENVIRONMENT} volumes: - ./source:/docs/source - docs_build:/docs/_build @@ -22,7 +22,7 @@ services: nginx: image: nginx:1.27-alpine3.20 - container_name: nginx + container_name: docs-nginx-${ENVIRONMENT} volumes: - docs_build:/var/www/html - ./nginx/conf.d:/etc/nginx/conf.d @@ -34,4 +34,4 @@ services: volumes: docs_build: - driver: local + driver: bridge From 50c62e339cfece7521d2796e17cce6cda02451f4 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:03:58 +0200 Subject: [PATCH 28/62] deploy on jenkins-11 --- docker-compose.yml | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/docker-compose.yml b/docker-compose.yml index 53d6b4bd8..621e42b30 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -5,6 +5,9 @@ services: context: . dockerfile: Dockerfile container_name: docs-sphinx-${ENVIRONMENT} + restart: unless-stopped + networks: + - docs-nginx volumes: - ./source:/docs/source - docs_build:/docs/_build @@ -21,8 +24,11 @@ services: # - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} nginx: - image: nginx:1.27-alpine3.20 + image: nginx:1.27.3-alpine3.20 container_name: docs-nginx-${ENVIRONMENT} + restart: unless-stopped + networks: + - docs-nginx volumes: - docs_build:/var/www/html - ./nginx/conf.d:/etc/nginx/conf.d @@ -30,8 +36,13 @@ services: ports: - ${LOCAL_PORT}:80 depends_on: - - sphinx + - docs-sphinx-${ENVIRONMENT} volumes: docs_build: + driver: local + +networks: + docs-nginx: + name: docs-nginx driver: bridge From e6713770f9e65ad7942fb052a99fd0bd96fe3424 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:04:52 +0200 Subject: [PATCH 29/62] deploy on jenkins-14 --- docker-compose.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docker-compose.yml b/docker-compose.yml index 621e42b30..3144905bd 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -36,7 +36,7 @@ services: ports: - ${LOCAL_PORT}:80 depends_on: - - docs-sphinx-${ENVIRONMENT} + - sphinx volumes: docs_build: From d8d082078b214bc7622a42bc4e9b5fe1692010a0 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:07:47 +0200 Subject: [PATCH 30/62] deploy on jenkins-15 --- deploy/Jenkinsfile | 8 -------- docker-compose.yml | 11 +++-------- 2 files changed, 3 insertions(+), 16 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 876fd0746..ae1bd7784 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,16 +2,8 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { - def envFile = """ -ENVIRONMENT=${env.ENVIRONMENT} -STAGING_HOST=${env.STAGING_HOST ?: ''} -PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} -LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} -""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ - echo "${envFile.trim()}" > .env && \ - cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ diff --git a/docker-compose.yml b/docker-compose.yml index 3144905bd..fd8ed9aff 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -12,16 +12,11 @@ services: - ./source:/docs/source - docs_build:/docs/_build environment: - - ENVIRONMENT=staging - - LOCAL_PORT=8080 + - ENVIRONMENT=${ENVIRONMENT} + - LOCAL_PORT=${LOCAL_PORT} - PRODUCTION_HOST=${PRODUCTION_HOST} - - STAGING_HOST=docs.bastion.talkable.com + - STAGING_HOST=${STAGING_HOST} - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} -# - ENVIRONMENT=${ENVIRONMENT} -# - LOCAL_PORT=${LOCAL_PORT} -# - PRODUCTION_HOST=${PRODUCTION_HOST} -# - STAGING_HOST=${STAGING_HOST} -# - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} nginx: image: nginx:1.27.3-alpine3.20 From 7ca13031a5d618e2a2958733dfff3878f6b7f3f7 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:11:25 +0200 Subject: [PATCH 31/62] deploy on jenkins-16 --- deploy/Jenkinsfile | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index ae1bd7784..5f17ff5b6 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,8 +2,16 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { + def envFile = """ +ENVIRONMENT=${env.ENVIRONMENT} +STAGING_HOST=${env.STAGING_HOST ?: ''} +PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} +LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} +""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ + echo "${envFile.trim()}" > .env && \ + cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ @@ -61,7 +69,6 @@ pipeline { // } steps { script { - sh 'printenv | sort' withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') } @@ -76,7 +83,6 @@ pipeline { } } steps { - sh 'printenv | sort' script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') From 38ae580935ee88e330308964d8fb2be975874631 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:18:40 +0200 Subject: [PATCH 32/62] deploy on jenkins-18 --- deploy/Jenkinsfile | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 5f17ff5b6..685860bd3 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,16 +2,8 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { - def envFile = """ -ENVIRONMENT=${env.ENVIRONMENT} -STAGING_HOST=${env.STAGING_HOST ?: ''} -PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} -LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} -""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ - echo "${envFile.trim()}" > .env && \ - cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ @@ -68,6 +60,7 @@ pipeline { // changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" // } steps { + sh "printenv | sort" script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') From 54de3e51f3171ea5ad22039201faf7a3296899c8 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:23:10 +0200 Subject: [PATCH 33/62] deploy on jenkins-20 --- deploy/Jenkinsfile | 9 ++++++++- docker-compose.yml | 8 ++------ 2 files changed, 10 insertions(+), 7 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 685860bd3..5f17ff5b6 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,8 +2,16 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { + def envFile = """ +ENVIRONMENT=${env.ENVIRONMENT} +STAGING_HOST=${env.STAGING_HOST ?: ''} +PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} +LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} +""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ + echo "${envFile.trim()}" > .env && \ + cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ @@ -60,7 +68,6 @@ pipeline { // changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" // } steps { - sh "printenv | sort" script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { dockerContainerUpdate('$HOST_ENV') diff --git a/docker-compose.yml b/docker-compose.yml index fd8ed9aff..60c96b465 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -8,15 +8,11 @@ services: restart: unless-stopped networks: - docs-nginx + env_file: + - ./.env volumes: - ./source:/docs/source - docs_build:/docs/_build - environment: - - ENVIRONMENT=${ENVIRONMENT} - - LOCAL_PORT=${LOCAL_PORT} - - PRODUCTION_HOST=${PRODUCTION_HOST} - - STAGING_HOST=${STAGING_HOST} - - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} nginx: image: nginx:1.27.3-alpine3.20 From 242f19977775079b330928429400a640e7e795b2 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:34:26 +0200 Subject: [PATCH 34/62] deploy on jenkins-22 --- deploy/Jenkinsfile | 8 -------- docker-compose.yml | 10 ++++++++-- 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 5f17ff5b6..650657d6c 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -2,16 +2,8 @@ // Upgrades or installs nginx containers for WordPress. def dockerContainerUpdate(String hostEnv) { - def envFile = """ -ENVIRONMENT=${env.ENVIRONMENT} -STAGING_HOST=${env.STAGING_HOST ?: ''} -PRODUCTION_HOST=${env.PRODUCTION_HOST ?: ''} -LOCAL_PORT=${env.LOCAL_PORT ?: '8080'} -""" sh """ssh -o StrictHostKeyChecking=no -i $SSH_KEY $SSH_USER@${hostEnv} 'cd talkable-docs/ && \ git checkout ${BRANCH_NAME} && git branch -a && git pull origin && \ - echo "${envFile.trim()}" > .env && \ - cat .env && \ docker-compose ls && docker ps && \ docker-compose down && docker-compose up -d --remove-orphans && \ docker-compose ls && docker ps' """ diff --git a/docker-compose.yml b/docker-compose.yml index 60c96b465..c18c3dac0 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -8,11 +8,17 @@ services: restart: unless-stopped networks: - docs-nginx - env_file: - - ./.env +# env_file: +# - ./.env volumes: - ./source:/docs/source - docs_build:/docs/_build + environment: + - ENVIRONMENT=${ENVIRONMENT} + - LOCAL_PORT=${LOCAL_PORT} + - PRODUCTION_HOST=${PRODUCTION_HOST} + - STAGING_HOST=${STAGING_HOST} + - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} nginx: image: nginx:1.27.3-alpine3.20 From b133cd888df6d24a96c5f8154c796e61e6543392 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:39:03 +0200 Subject: [PATCH 35/62] deploy on jenkins-23 --- deploy/Jenkinsfile | 20 -------------------- 1 file changed, 20 deletions(-) diff --git a/deploy/Jenkinsfile b/deploy/Jenkinsfile index 650657d6c..b3c807eba 100644 --- a/deploy/Jenkinsfile +++ b/deploy/Jenkinsfile @@ -17,7 +17,6 @@ pipeline { GITHUB_ACC = credentials('talkable-bot-deploy-token') JENKINS_BUILD_USER_ID = "" LOCAL_PORT="8080" -// CHANGESET_REGEXP = "^nginx/.*" } triggers { pollSCM('* * * * *') // Polls SCM every minute @@ -55,25 +54,6 @@ pipeline { } } stage('Build') { -// when { -// beforeAgent true -// changeset pattern: "${CHANGESET_REGEXP}", comparator: "REGEXP" -// } - steps { - script { - withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { - dockerContainerUpdate('$HOST_ENV') - } - } - } - } - stage('Manual-build') { - when { - not - { - triggeredBy 'SCMTrigger' - } - } steps { script { withCredentials([sshUserPrivateKey(credentialsId: "${JENKINS_CRED_ID}", keyFileVariable: 'SSH_KEY', usernameVariable: 'SSH_USER')]) { From 344157fa67d2a540af95ddbe3a595c6a228f0d3e Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:43:47 +0200 Subject: [PATCH 36/62] deploy on jenkins-23 --- docker-compose.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/docker-compose.yml b/docker-compose.yml index c18c3dac0..d8ee3e51a 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,6 +1,7 @@ services: sphinx: + image: talkable-docs-sphinx:${BUILD_ID} build: context: . dockerfile: Dockerfile From 56eae6cb499f8462469e9a070f645b73ec524429 Mon Sep 17 00:00:00 2001 From: "sergii.kostiuk" Date: Tue, 28 Jan 2025 01:45:55 +0200 Subject: [PATCH 37/62] deploy on jenkins-24 --- docker-compose.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docker-compose.yml b/docker-compose.yml index d8ee3e51a..535e194f0 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,7 +1,7 @@ services: sphinx: - image: talkable-docs-sphinx:${BUILD_ID} + image: talkable-docs-sphinx:${BUILD_ID:-tagname} build: context: . dockerfile: Dockerfile From 992d40d77cd560ad668134a8031deccd50b82b7e Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 11:25:31 +0300 Subject: [PATCH 38/62] install and run nginx and sphinx within the same container --- Dockerfile | 19 ++++- docker-compose.yml | 25 +++---- nginx/{conf.d => http.d}/default.conf | 3 +- nginx/nginx.conf | 104 ++++++++++++++++++++++++++ 4 files changed, 132 insertions(+), 19 deletions(-) rename nginx/{conf.d => http.d}/default.conf (98%) create mode 100644 nginx/nginx.conf diff --git a/Dockerfile b/Dockerfile index 5dba08b0b..b6aace6c5 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,14 +1,27 @@ FROM python:3.13-alpine3.21 -# Install dependencies WORKDIR /docs + ADD requirements.txt /docs RUN python3 -m pip install -r requirements.txt +RUN apk add --no-cache \ + # gcc \ + # musl-dev \ + # libffi-dev \ + # openssl-dev \ + # make + nginx + +# Create necessary directories for Nginx +RUN mkdir -p /run/nginx + CMD if [ "$ENVIRONMENT" = "development" ]; then \ echo "Running Sphinx in Development mode"; \ - sphinx-autobuild -b dirhtml /docs/source /docs/_build; \ + sphinx-autobuild -b dirhtml --port 8000 --host 0.0.0.0 /docs/source /docs/_build; \ else \ echo "Running Sphinx in Staging/Production mode"; \ - sphinx-build -b dirhtml /docs/source /docs/_build; \ + sphinx-build -b dirhtml /docs/source /var/www/html; \ + echo "Starting Nginx"; \ + nginx -g "daemon off;"; \ fi diff --git a/docker-compose.yml b/docker-compose.yml index 4a23ddfe9..c9c2c6241 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,31 +1,26 @@ services: - sphinx: + talkable-docs: build: context: . dockerfile: Dockerfile - container_name: sphinx + container_name: talkable-docs volumes: + # Sphinx - ./source:/docs/source - - docs_build:/docs/_build + # Nginx + - ./nginx/http.d:/etc/nginx/http.d + - ./nginx/nginx.conf:/etc/nginx/nginx.conf:ro + - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt + # Build volume to keep the docs available + - docs_build:/var/www/html environment: - ENVIRONMENT=${ENVIRONMENT} - LOCAL_PORT=${LOCAL_PORT} - PRODUCTION_HOST=${PRODUCTION_HOST} - STAGING_HOST=${STAGING_HOST} - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} - - nginx: - image: nginx:1.27-alpine3.20 - container_name: nginx - volumes: - - docs_build:/var/www/html - - ./nginx/conf.d:/etc/nginx/conf.d - - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt ports: - - ${LOCAL_PORT}:80 - depends_on: - - sphinx + - ${LOCAL_PORT}:8000 volumes: docs_build: - driver: local diff --git a/nginx/conf.d/default.conf b/nginx/http.d/default.conf similarity index 98% rename from nginx/conf.d/default.conf rename to nginx/http.d/default.conf index 5eb1145ca..a01a45c23 100644 --- a/nginx/conf.d/default.conf +++ b/nginx/http.d/default.conf @@ -1,5 +1,6 @@ server { - listen 80; + + listen 8000; server_name _; root /var/www/html; diff --git a/nginx/nginx.conf b/nginx/nginx.conf new file mode 100644 index 000000000..f4a3741b6 --- /dev/null +++ b/nginx/nginx.conf @@ -0,0 +1,104 @@ +# /etc/nginx/nginx.conf + +user nginx; + +# Set number of worker processes automatically based on number of CPU cores. +worker_processes auto; + +# Enables the use of JIT for regular expressions to speed-up their processing. +pcre_jit on; + +# Configures default error logger. +# error_log /var/log/nginx/error.log warn; +error_log /dev/stderr info; + +# Includes files with directives to load dynamic modules. +include /etc/nginx/modules/*.conf; + +# Include files with config snippets into the root context. +include /etc/nginx/conf.d/*.conf; + +events { + # The maximum number of simultaneous connections that can be opened by + # a worker process. + worker_connections 1024; +} + +http { + # Includes mapping of file name extensions to MIME types of responses + # and defines the default type. + include /etc/nginx/mime.types; + default_type application/octet-stream; + + # Name servers used to resolve names of upstream servers into addresses. + # It's also needed when using tcpsocket and udpsocket in Lua modules. + #resolver 1.1.1.1 1.0.0.1 [2606:4700:4700::1111] [2606:4700:4700::1001]; + + # Don't tell nginx version to the clients. Default is 'on'. + server_tokens off; + + # Specifies the maximum accepted body size of a client request, as + # indicated by the request header Content-Length. If the stated content + # length is greater than this size, then the client receives the HTTP + # error code 413. Set to 0 to disable. Default is '1m'. + client_max_body_size 1m; + + # Sendfile copies data between one FD and other from within the kernel, + # which is more efficient than read() + write(). Default is off. + sendfile on; + + # Causes nginx to attempt to send its HTTP response head in one packet, + # instead of using partial frames. Default is 'off'. + tcp_nopush on; + + + # Enables the specified protocols. Default is TLSv1 TLSv1.1 TLSv1.2. + # TIP: If you're not obligated to support ancient clients, remove TLSv1.1. + ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3; + + # Path of the file with Diffie-Hellman parameters for EDH ciphers. + # TIP: Generate with: `openssl dhparam -out /etc/ssl/nginx/dh2048.pem 2048` + #ssl_dhparam /etc/ssl/nginx/dh2048.pem; + + # Specifies that our cipher suits should be preferred over client ciphers. + # Default is 'off'. + ssl_prefer_server_ciphers on; + + # Enables a shared SSL cache with size that can hold around 8000 sessions. + # Default is 'none'. + ssl_session_cache shared:SSL:2m; + + # Specifies a time during which a client may reuse the session parameters. + # Default is '5m'. + ssl_session_timeout 1h; + + # Disable TLS session tickets (they are insecure). Default is 'on'. + ssl_session_tickets off; + + + # Enable gzipping of responses. + #gzip on; + + # Set the Vary HTTP header as defined in the RFC 2616. Default is 'off'. + gzip_vary on; + + + # Helper variable for proxying websockets. + map $http_upgrade $connection_upgrade { + default upgrade; + '' close; + } + + + # Specifies the main log format. + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + + # Sets the path, format, and configuration for a buffered log write. + # access_log /var/log/nginx/access.log main; + access_log /dev/stdout; + + # Includes virtual hosts configs. + include /etc/nginx/http.d/*.conf; +} From 99e7e8772c67c66471bd9724d9c7f2cc1d64e6e1 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 11:42:32 +0300 Subject: [PATCH 39/62] docker compose refactoring --- docker-compose.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docker-compose.yml b/docker-compose.yml index dbaa8cca4..1293a681b 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -3,7 +3,8 @@ services: build: context: . dockerfile: Dockerfile - container_name: talkable-docs + container_name: docs-sphinx-${ENVIRONMENT} + restart: unless-stopped volumes: # Sphinx - ./source:/docs/source From 301e3abf5a0e17c455df8b4c5ac27dfb279fb9c0 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 14:41:00 +0300 Subject: [PATCH 40/62] update the readme files to sync with one container design (python instead of nginx and python) --- README-maintainer.md | 49 +++----------------------------------------- README.md | 12 +++-------- 2 files changed, 6 insertions(+), 55 deletions(-) diff --git a/README-maintainer.md b/README-maintainer.md index e9891b437..c53e39e09 100644 --- a/README-maintainer.md +++ b/README-maintainer.md @@ -15,7 +15,6 @@ 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) @@ -60,16 +59,10 @@ The goal is to update `requirements.txt` with the latest versions of dependencie 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: + If the documentation fails to load, check the container logs: ```bash - docker logs -f sphinx - ``` - - ...and the Nginx logs: - - ```bash - docker logs -f nginx + docker logs -f docs-sphinx-development ``` 3. Test and freeze `requirements.txt`. @@ -83,7 +76,7 @@ The goal is to update `requirements.txt` with the latest versions of dependencie Save the dependencies with the following command: ```bash - docker exec sphinx pip freeze > requirements.txt + docker exec docs-sphinx-development pip freeze > requirements.txt ``` 4. Stop the containers. @@ -98,42 +91,6 @@ The goal is to update `requirements.txt` with the latest versions of dependencie 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. diff --git a/README.md b/README.md index f89a948fb..1b4939068 100644 --- a/README.md +++ b/README.md @@ -48,7 +48,7 @@ The repository consists of the following branches: Run the command: ```bash - docker compose up -d + docker compose up -d --build ``` 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. @@ -118,16 +118,10 @@ name "Talkable Section". 1. Ensure you are using the correct port number and protocol. The port number should match the value provided in `.env` as `LOCAL_PORT`. -2. Check `nginx` logs: +2. Check logs: ```bash - docker logs -f nginx - ``` - -3. Check `Sphinx` logs: - - ```bash - docker logs -f sphinx + docker logs -f docs-sphinx-development ``` ## Links From 95a2602fd1d304d235b36ac67217eb1807875253 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 15:44:39 +0300 Subject: [PATCH 41/62] [PR-23789] "empty screen on empty search" fix --- nginx/http.d/default.conf | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/nginx/http.d/default.conf b/nginx/http.d/default.conf index a01a45c23..d282b08a7 100644 --- a/nginx/http.d/default.conf +++ b/nginx/http.d/default.conf @@ -56,6 +56,14 @@ server { try_files $uri =404; } + # Redirect requests starting with /search/?q=& to / + # Fix of the Furo theme bug described here https://talkable.atlassian.net/browse/PR-23789 + location /search/ { + if ($arg_q = "") { + return 301 /; + } + } + # Serve robots.txt from a specific file location = /robots.txt { alias /var/www/robots.txt; From 8edf02455c15b21293a174c008228a49a0262152 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 15:56:17 +0300 Subject: [PATCH 42/62] legacy requirements-dev.txt cleanup --- requirements-dev.txt | 5 ----- 1 file changed, 5 deletions(-) delete mode 100644 requirements-dev.txt 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 From 5755c7e8894a0f5439832168c89cf439b973cd64 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 16:46:30 +0300 Subject: [PATCH 43/62] lists fixes --- source/advanced_features/subscribing_to_events.rst | 6 ++---- source/advanced_features/tremendous.rst | 1 - source/partials/campaign_placements_description.rst | 7 +++---- 3 files changed, 5 insertions(+), 9 deletions(-) diff --git a/source/advanced_features/subscribing_to_events.rst b/source/advanced_features/subscribing_to_events.rst index c90ba5129..61ed5a2c9 100644 --- a/source/advanced_features/subscribing_to_events.rst +++ b/source/advanced_features/subscribing_to_events.rst @@ -31,9 +31,8 @@ Knowing the container `name` attribute of the iframe is `talkable-offer-iframe` }); Notice two arguments passed in the callback: - - 1. `data` object — the data passed by the iframe upon firing the event - 2. `iframe` object — iframe’s `HTML DOM reference`_ + - `data` object — the data passed by the iframe upon firing the event + - `iframe` object — iframe’s `HTML DOM reference`_ .. _HTML DOM reference: https://www.w3schools.com/jsref/dom_obj_frame.asp @@ -41,7 +40,6 @@ Iframe Events List ------------------ Talkable campaigns are equipped with the following set of events: - 1. `offer_loaded` — Talkable iframe DOM ready event which is in fact the very first event that you can use to determine if the iframe is loaded. 2. `responsive_iframe_height` — fires every time iframe size gets changed. You can use it to detect changes in iframe size. You can read more :ref:`here `. Previously named as ``curebit_offer_iframe_broadcast``. diff --git a/source/advanced_features/tremendous.rst b/source/advanced_features/tremendous.rst index 1e0fb2a75..2ab74ea04 100644 --- a/source/advanced_features/tremendous.rst +++ b/source/advanced_features/tremendous.rst @@ -10,7 +10,6 @@ Tremendous Integration Reward advocates with gift cards from various retailers provided by Tremendous. Tremendous integration consists of several parts: - 1. A campaign in Tremendous with the selected rewards for customers. 3. A "Tremendous" application installed from Talkable App Store. This application is responsible diff --git a/source/partials/campaign_placements_description.rst b/source/partials/campaign_placements_description.rst index 54a902756..f4a790c0f 100644 --- a/source/partials/campaign_placements_description.rst +++ b/source/partials/campaign_placements_description.rst @@ -39,10 +39,9 @@ A Placement can have a set of criteria defined based on :ref:`custom properties If all of the criteria are met, the placement will show a campaign. The criteria can be defined using one of the following operators: - -* `eq` -* `not_eq` -* `regex` (performs a case-insensitive match) + - `eq` + - `not_eq` + - `regex` (performs a case-insensitive match) .. note:: Custom Property Criteria require integration version 5.1.4 or higher to work. From 9d3883ad88c99b08f7ca62592ccd6acff608ae30 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 18:31:31 +0300 Subject: [PATCH 44/62] [PR-23796] add missing "," --- source/partials/purchase_script.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/source/partials/purchase_script.rst b/source/partials/purchase_script.rst index c862b358c..82f83745e 100644 --- a/source/partials/purchase_script.rst +++ b/source/partials/purchase_script.rst @@ -10,7 +10,7 @@ coupon_code: '', // Coupon code that was used at checkout (pass multiple as an array). Example: 'SAVE20' currency_iso_code: '', // Required for multi-currency sites. Example: 'USD' shipping_zip: '', // Used for fraud protection by address. Example: '02222' - shipping_address: '' // Full address of the order, make sure to strictly follow a format: 'Apt #, Street address, City, State, ZIP, Country' + shipping_address: '', // Full address of the order, make sure to strictly follow a format: 'Apt #, Street address, City, State, ZIP, Country' segment1: '', // Segment 1: Represents custom segment (e.g., location, age group, source channel, platform, gender, interests). segment2: '', // Segment 2: Represents custom segment (e.g., location, age group, source channel, platform, gender, interests). segment3: '', // Segment 3: Represents custom segment (e.g., location, age group, source channel, platform, gender, interests). From 3f00f08c5b408c5458a04663d2a3b16042c5926c Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 19:30:31 +0300 Subject: [PATCH 45/62] Revert "install and run nginx and sphinx within the same container" This reverts commit 992d40d77cd560ad668134a8031deccd50b82b7e. --- Dockerfile | 19 +---- docker-compose.yml | 24 +++--- nginx/{http.d => conf.d}/default.conf | 3 +- nginx/nginx.conf | 104 -------------------------- 4 files changed, 18 insertions(+), 132 deletions(-) rename nginx/{http.d => conf.d}/default.conf (98%) delete mode 100644 nginx/nginx.conf diff --git a/Dockerfile b/Dockerfile index b6aace6c5..5dba08b0b 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,27 +1,14 @@ FROM python:3.13-alpine3.21 +# Install dependencies WORKDIR /docs - ADD requirements.txt /docs RUN python3 -m pip install -r requirements.txt -RUN apk add --no-cache \ - # gcc \ - # musl-dev \ - # libffi-dev \ - # openssl-dev \ - # make - nginx - -# Create necessary directories for Nginx -RUN mkdir -p /run/nginx - CMD if [ "$ENVIRONMENT" = "development" ]; then \ echo "Running Sphinx in Development mode"; \ - sphinx-autobuild -b dirhtml --port 8000 --host 0.0.0.0 /docs/source /docs/_build; \ + sphinx-autobuild -b dirhtml /docs/source /docs/_build; \ else \ echo "Running Sphinx in Staging/Production mode"; \ - sphinx-build -b dirhtml /docs/source /var/www/html; \ - echo "Starting Nginx"; \ - nginx -g "daemon off;"; \ + sphinx-build -b dirhtml /docs/source /docs/_build; \ fi diff --git a/docker-compose.yml b/docker-compose.yml index 1293a681b..b7ec791fd 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,27 +1,31 @@ services: - talkable-docs: + sphinx: build: context: . dockerfile: Dockerfile container_name: docs-sphinx-${ENVIRONMENT} restart: unless-stopped volumes: - # Sphinx - ./source:/docs/source - # Nginx - - ./nginx/http.d:/etc/nginx/http.d - - ./nginx/nginx.conf:/etc/nginx/nginx.conf:ro - - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt - # Build volume to keep the docs available - - docs_build:/var/www/html + - docs_build:/docs/_build environment: - ENVIRONMENT=${ENVIRONMENT} - LOCAL_PORT=${LOCAL_PORT} - PRODUCTION_HOST=${PRODUCTION_HOST} - STAGING_HOST=${STAGING_HOST} - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} + + nginx: + image: nginx:1.27-alpine3.20 + container_name: nginx + volumes: + - docs_build:/var/www/html + - ./nginx/conf.d:/etc/nginx/conf.d + - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt ports: - - ${LOCAL_PORT}:8000 + - ${LOCAL_PORT}:80 + depends_on: + - sphinx volumes: - docs_build: \ No newline at end of file + docs_build: diff --git a/nginx/http.d/default.conf b/nginx/conf.d/default.conf similarity index 98% rename from nginx/http.d/default.conf rename to nginx/conf.d/default.conf index d282b08a7..5788ccada 100644 --- a/nginx/http.d/default.conf +++ b/nginx/conf.d/default.conf @@ -1,6 +1,5 @@ server { - - listen 8000; + listen 80; server_name _; root /var/www/html; diff --git a/nginx/nginx.conf b/nginx/nginx.conf deleted file mode 100644 index f4a3741b6..000000000 --- a/nginx/nginx.conf +++ /dev/null @@ -1,104 +0,0 @@ -# /etc/nginx/nginx.conf - -user nginx; - -# Set number of worker processes automatically based on number of CPU cores. -worker_processes auto; - -# Enables the use of JIT for regular expressions to speed-up their processing. -pcre_jit on; - -# Configures default error logger. -# error_log /var/log/nginx/error.log warn; -error_log /dev/stderr info; - -# Includes files with directives to load dynamic modules. -include /etc/nginx/modules/*.conf; - -# Include files with config snippets into the root context. -include /etc/nginx/conf.d/*.conf; - -events { - # The maximum number of simultaneous connections that can be opened by - # a worker process. - worker_connections 1024; -} - -http { - # Includes mapping of file name extensions to MIME types of responses - # and defines the default type. - include /etc/nginx/mime.types; - default_type application/octet-stream; - - # Name servers used to resolve names of upstream servers into addresses. - # It's also needed when using tcpsocket and udpsocket in Lua modules. - #resolver 1.1.1.1 1.0.0.1 [2606:4700:4700::1111] [2606:4700:4700::1001]; - - # Don't tell nginx version to the clients. Default is 'on'. - server_tokens off; - - # Specifies the maximum accepted body size of a client request, as - # indicated by the request header Content-Length. If the stated content - # length is greater than this size, then the client receives the HTTP - # error code 413. Set to 0 to disable. Default is '1m'. - client_max_body_size 1m; - - # Sendfile copies data between one FD and other from within the kernel, - # which is more efficient than read() + write(). Default is off. - sendfile on; - - # Causes nginx to attempt to send its HTTP response head in one packet, - # instead of using partial frames. Default is 'off'. - tcp_nopush on; - - - # Enables the specified protocols. Default is TLSv1 TLSv1.1 TLSv1.2. - # TIP: If you're not obligated to support ancient clients, remove TLSv1.1. - ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3; - - # Path of the file with Diffie-Hellman parameters for EDH ciphers. - # TIP: Generate with: `openssl dhparam -out /etc/ssl/nginx/dh2048.pem 2048` - #ssl_dhparam /etc/ssl/nginx/dh2048.pem; - - # Specifies that our cipher suits should be preferred over client ciphers. - # Default is 'off'. - ssl_prefer_server_ciphers on; - - # Enables a shared SSL cache with size that can hold around 8000 sessions. - # Default is 'none'. - ssl_session_cache shared:SSL:2m; - - # Specifies a time during which a client may reuse the session parameters. - # Default is '5m'. - ssl_session_timeout 1h; - - # Disable TLS session tickets (they are insecure). Default is 'on'. - ssl_session_tickets off; - - - # Enable gzipping of responses. - #gzip on; - - # Set the Vary HTTP header as defined in the RFC 2616. Default is 'off'. - gzip_vary on; - - - # Helper variable for proxying websockets. - map $http_upgrade $connection_upgrade { - default upgrade; - '' close; - } - - - # Specifies the main log format. - log_format main '$remote_addr - $remote_user [$time_local] "$request" ' - '$status $body_bytes_sent "$http_referer" ' - '"$http_user_agent" "$http_x_forwarded_for"'; - - # Sets the path, format, and configuration for a buffered log write. - # access_log /var/log/nginx/access.log main; - access_log /dev/stdout; - - # Includes virtual hosts configs. - include /etc/nginx/http.d/*.conf; -} From a7ff19b6e812dc39ea47a07df525dc5b7ee76c56 Mon Sep 17 00:00:00 2001 From: en-ver Date: Tue, 28 Jan 2025 19:32:25 +0300 Subject: [PATCH 46/62] add ENVIRONMENT variable to nginx container name --- docker-compose.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docker-compose.yml b/docker-compose.yml index b7ec791fd..4c4f50605 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -17,7 +17,7 @@ services: nginx: image: nginx:1.27-alpine3.20 - container_name: nginx + container_name: nginx-${ENVIRONMENT} volumes: - docs_build:/var/www/html - ./nginx/conf.d:/etc/nginx/conf.d From 2872eb89833a3dd422973202c5da9598f90cd048 Mon Sep 17 00:00:00 2001 From: Vitalii Oliinyk Date: Tue, 28 Jan 2025 18:42:43 +0200 Subject: [PATCH 47/62] [PR-23724]: updated views --- .../converting_into_localization.rst | 65 +++++++++---------- .../shopify_coupons_auto_apply.rst | 15 ++--- .../shopify_coupons_auto_sync.rst | 6 -- .../shopify_purchase_syncing.rst | 10 ++- source/advanced_features/single_sign_on.rst | 1 - .../white_labeling/godaddy.rst | 11 +--- .../white_labeling/shopify.rst | 8 +-- source/android_sdk/api.rst | 10 +-- source/android_sdk/getting_started.rst | 2 +- source/campaigns/editor/partials.rst | 26 ++++---- .../campaigns/views/friend_rewards_paid.rst | 8 +-- .../campaigns/views/notifier_offers_email.rst | 7 +- .../views/notifier_offers_share_via_email.rst | 2 +- ...tifier_offers_share_via_email_reminder.rst | 6 +- .../custom_app.rst | 2 - .../klaviyo.rst | 1 - .../custom/integration_components.rst | 1 - .../ecommerce_platforms/magento/magento1.rst | 1 - .../ecommerce_platforms/magento/magento2.rst | 7 -- .../ecommerce_platforms/shopify.rst | 61 +++-------------- source/integration/loyalty.rst | 6 +- source/integration/loyalty/custom_app.rst | 1 - .../loyalty/integration_components.rst | 1 - .../loyalty/shopify_integration.rst | 13 +--- source/ios_sdk/advanced.rst | 37 +++++------ source/ios_sdk/getting_started.rst | 23 ++++--- source/ios_sdk/social_sharing.rst | 5 +- source/web_hooks/check_unsubscribe.rst | 9 ++- source/web_hooks/post_share.rst | 20 ++---- 29 files changed, 132 insertions(+), 233 deletions(-) diff --git a/source/advanced_features/converting_into_localization.rst b/source/advanced_features/converting_into_localization.rst index da30f6484..d6de47f44 100644 --- a/source/advanced_features/converting_into_localization.rst +++ b/source/advanced_features/converting_into_localization.rst @@ -10,7 +10,6 @@ Converting Into Localization If you don’t know what is Localization inside Talkable :ref:`read this article `. Few benefits you get out of using Localizations: - 1. It is extremely easy to set up an AB test if your copy is coded up as a Localization. 2. For non-technical people it is easier to change the copy inside Localization Editor because they are afraid to code. @@ -66,23 +65,23 @@ It is highly unlikely that your campaign is not equipped with Subject line as a Subject line is unique because its default value is set on the Advocate Signup/Share Page along with other email sharing fields (email body, reminder checkbox value), not Friend Share email as it might sounded logical. Here is the plan: - 1. Create Subject Line as a Localization on the Advocate Signup/Share Page, provide its default value, it will be used inside `value` attribute of the "subject" form field: +1. Create Subject Line as a Localization on the Advocate Signup/Share Page, provide its default value, it will be used inside `value` attribute of the "subject" form field: - .. image:: /_static/img/advanced_features/subject_line_setup_inside_share_page.png + .. image:: /_static/img/advanced_features/subject_line_setup_inside_share_page.png - .. code-block:: html + .. code-block:: html - This code creates new Localization named "Friend share email subject" that you are able to change on the Advocate Signup/Share Page. + This code creates new Localization named "Friend share email subject" that you are able to change on the Advocate Signup/Share Page. - 2. Navigate to Friend Share email → Extra fields to see Email Subject field: +2. Navigate to Friend Share email → Extra fields to see Email Subject field: - .. image:: /_static/img/advanced_features/campaign_editor_subject.png + .. image:: /_static/img/advanced_features/campaign_editor_subject.png - 3. Put the following code in there: +3. Put the following code in there: - .. code-block:: liquid + .. code-block:: liquid {% if custom_message_subject == blank %} {{ 'friend_share_email_subject' | localize }} @@ -90,20 +89,20 @@ Subject line is unique because its default value is set on the Advocate Signup/S {{ custom_message_subject }} {% endif %} - The code snippet above checks if the Advocate provided any Subject at all. If not we take default Subject copy so Friend Share email does not come with a blank subject. + The code snippet above checks if the Advocate provided any Subject at all. If not we take default Subject copy so Friend Share email does not come with a blank subject. Color As Localization --------------------- Another example would be localizing font color of a headline, all copy at once, or a background color of a button. You can use `color` trait of a Localization for that. - 1. Navigate to HTML & CSS editor of the View you want to add a color Localization on: +1. Navigate to HTML & CSS editor of the View you want to add a color Localization on: - .. image:: /_static/img/advanced_features/campaign_navigation_html_css_editor.png + .. image:: /_static/img/advanced_features/campaign_navigation_html_css_editor.png - 2. At the very bottom of the HTML area add ```` tag with CSS that will override default styling of the element you want to localize: +2. At the very bottom of the HTML area add ```` tag with CSS that will override default styling of the element you want to localize: - .. code-block:: text + .. code-block:: text .button { background-color: {{ "advocate_share_page_button_color" | localize: "#f94d08", trait: "color" }}; @@ -112,22 +111,22 @@ Another example would be localizing font color of a headline, all copy at once, In the code example above we created new Color Localization with default HEX color `#f94d08` which is used for `background-color` and `border-color` CSS properties of `.button` selector. Whenever you set a new color inside Campaign Editor it will be changed across both places because we’re using the same Localization identifier in both places. - 3. New Color Localization appears under "Color" tab inside Campaign Editor: +3. New Color Localization appears under "Color" tab inside Campaign Editor: - .. image:: /_static/img/advanced_features/editor_colors_tab.png + .. image:: /_static/img/advanced_features/editor_colors_tab.png Image As Localization --------------------- Localizing Image asset can be handy if you want to AB test it. Here is how to do that: - 1. Navigate to HTML & CSS editor of the View you want to add a color Localization on: +1. Navigate to HTML & CSS editor of the View you want to add a color Localization on: - .. image:: /_static/img/advanced_features/campaign_navigation_html_css_editor.png + .. image:: /_static/img/advanced_features/campaign_navigation_html_css_editor.png - 2. Inside HTML area find an image you want to localize. An image can be either within CSS or within HTML area (``, inline styles, etc.). If the image is set within CSS you need to extract it into HTML area using inline styles: +2. Inside HTML area find an image you want to localize. An image can be either within CSS or within HTML area (``, inline styles, etc.). If the image is set within CSS you need to extract it into HTML area using inline styles: - .. code-block:: text + .. code-block:: text
... @@ -135,9 +134,9 @@ Localizing Image asset can be handy if you want to AB test it. Here is how to do In the example above `share_page_background` is a name of an Image Localization. `share-page-background.jpg` is a name of an Asset (Files tab within HTML & CSS Editor). - 3. Now we can see newly created Image Localization under "Images" tab: +3. Now we can see newly created Image Localization under "Images" tab: - .. image:: /_static/img/advanced_features/editor_images_tab.png + .. image:: /_static/img/advanced_features/editor_images_tab.png Custom Option (Configuration) Localization ------------------------------------------ @@ -146,17 +145,17 @@ In addition to localizing Images, Colors, and static copy Talkable allows you to An example can be to create an AB test for Equal Emphasis (all 3 sharing channels look visually equal) vs. Email Emphasis where email sharing form stands out: - .. image:: /_static/img/advanced_features/share_page_equal_emphasis.png - :height: 250 px + .. image:: /_static/img/advanced_features/share_page_equal_emphasis.png + :height: 250 px - .. image:: /_static/img/advanced_features/share_page_email_emphasis.png - :height: 250 px + .. image:: /_static/img/advanced_features/share_page_email_emphasis.png + :height: 250 px In order to chieve this AB test we need to - 1. Build two separate layouts using CSS cascades to style all nested children within a container block that holds all the content: +1. Build two separate layouts using CSS cascades to style all nested children within a container block that holds all the content: - .. code-block:: text + .. code-block:: text {% assign share_page_layout = "share_page_layout" | localize: "Equal Emphasis", "Email Emphasis" %}