Skip to main content

Flask extension to integrate discourse content generated to docs to your website.

Project description

canonicalwebteam.discourse

Flask extension to integrate discourse content generated to docs to your website. This project was previously named discourse_docs.

Writing documentation

Documentation for how to write documentation pages in Discourse for consumption by this module and how to configure the website to use the module can be found in the Canonical discourse.

Example Flask template for documentation pages can be found in examples folder. Please refer to the README in that folder for more information.

Install

Install the project with pip: pip install canonicalwebteam.discourse

You can add the extension on your project as follows, replacing, at least, base_url and index_topic_id with your own settings:

import talisker.requests
from canonicalwebteam.discourse import DiscourseAPI, Tutorials, TutorialParser

app = Flask("myapp")
session = talisker.requests.get_session()

discourse = Tutorials(
    parser=TutorialParser(
        api=DiscourseAPI(
            base_url="https://forum.example.com/", session=session
        ),
        index_topic_id=321,
        url_prefix="/docs",
    ),
    document_template="docs/document.html",
    url_prefix="/docs",
)
discourse.init_app(app)

Once this is added you will need to add the file document.html to your template folder.

Local development

For local development, it's best to test this module with one of our website projects like ubuntu.com. For more information, follow this guide (internal only).

Running tests, linting and formatting

Tests can be run with Tox:

pip3 install tox  # Install tox
tox               # Run tests
tox -e lint       # Check the format of Python code
tox -e format     # Reformat the Python code

Instructions for Engage pages extension

Because you are viewing a protected topic, you must provide api_key and api_username. You also need an index topic id, which you can get from discourse.ubuntu.com. Your index topic must contain a metadata section. Visit the EngageParser for more information about the structure. You are encouraged to use an blueprint name that does not collide with existent blueprints. The templates must match the ones provided in the parameters indicated.

Here is an example of an implementation:

engage_pages = EngagePages(
    api=DiscourseAPI(
        base_url="https://discourse.ubuntu.com/",
        session=session,
        get_topics_query_id=14,
        api_key=DISCOURSE_API_KEY, # replace with your API key
        api_username=DISCOURSE_API_USERNAME, # replace with correspoding username
    ),
    category_id=51,
    page_type="engage-pages", # one of ["engage-pages", "takeovers"]
    exclude_topics=[] # this is a list of topic ids that we want to exclude from Markdown error checks
    additional_metadata_validation=[] # list of additional keys in the metadata table that you want to validate existence for e.g. language
)

In your project, you need to create your own views:

app.add_url_rule(
    "/engage", view_func=build_engage_index(engage_pages)
)

app.add_url_rule(
    "/engage/<path>", view_func=single_engage_page(engage_pages)
)
  • Where build_engage_index would be your view for the list of engage pages, which you can get by using the method EngagagePages(args).get_index()
  • While single_engage_page would be your single engage pages view, which you can get using EngagePages(args).get_engage_page(path)

Similarly for takeovers, you just need to pass page_type="takeovers".

  • To get a list of takeovers EngagePages(args).get_index() also.
  • To get a list of active takeovers EngagePages(args).parse_active_takeovers().

Pagination

  • get_index provides two additional arguments limit and offset, to provide pagination functionality. They default to 50 and 0 respectively.
  • If you want to get all engage pages, which in the case of some sites like jp.ubuntu.com there are not that many, you can pass limit=-1
  • Use MaxLimitError in the exceptions.py to handle excessive limit. By default, it will raise an error when it surpasses 500

Tag filtering for Engage Pages

get_index and the underlying API methods accept a tag parameter that supports filtering by one or more tags using OR logic:

  • Single tag (string, legacy): engage_pages.get_index(tag_value="osm")
  • Multiple tags, OR logic (list): engage_pages.get_index(tag_value=["osm", "gsi"])
  • No tag filter: omit the parameter, pass None, or pass an empty list

Tags are matched case-insensitively and duplicates are removed automatically. The same str | list[str] interface applies to the tag parameter of DiscourseAPI.get_engage_pages_by_tag() and the tag_value parameter of DiscourseAPI.get_engage_pages_by_param().

Instructions for Category class usage

This works similar to the other class but exposes some specific functions that can be run on the index topic and the category as a whole.

It exposes a some APIs that can then be called from within a view func for processing.

Here is an example of the implementation:

security_vulnerabilities = Category(
    parser=CategoryParser(
        api=discourse_api,
        index_topic_id=53193,
        url_prefix="/security/vulnerabilities",
    ),
    category_id=308,
)

The security_vulnerabilities object exposes the following APIs:

  • get_topic(path): Fetches a single topic using its URL (path).
  • get_category_index_metadata(data_name): Retrieves metadata for the category index. You can optionally specify a data_name to get data for just one table.
  • get_topics_in_category(): Retrieves all topics within the currently active category.
  • get_category_events(limit=100, offset=0): Retrieves all future events in a category. Requires the Discourse Events plugin to be installed on the instance.

Instructions for Events class usage

This class provides functionality for managing and parsing events from Discourse topics, particularly useful for event-driven websites that need to display upcoming events, featured events, and event categories. It relies on the plugin, Discourse Calendar.

It exposes APIs that can be called from within a view function for processing event data.

Here is an example of the implementation:

events = Events(
    parser=EventsParser(
        api=discourse_api,
        index_topic_id=12345,
        url_prefix="/events",
    ),
    category_id=25,
)

The events object exposes the following APIs:

  • get_events(): Fetches all future events from the target Discourse instance.
  • get_featured_events(target_tag="featured-event"): Retrieves all events with a given tagrte tag, defaults to "featured-event"

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

canonicalwebteam_discourse-7.5.0.tar.gz (47.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

canonicalwebteam_discourse-7.5.0-py3-none-any.whl (51.8 kB view details)

Uploaded Python 3

File details

Details for the file canonicalwebteam_discourse-7.5.0.tar.gz.

File metadata

File hashes

Hashes for canonicalwebteam_discourse-7.5.0.tar.gz
Algorithm Hash digest
SHA256 78bf6945481be27a637cff5167f53f3a1d1a1df83ff9d83a2b9c35dbe90f5100
MD5 c7547c032b952bff5ea49091a79ea74c
BLAKE2b-256 1e27f6a20b2b32bb88712b7872322a5e8e65a507d15efe8a8e43c40908c29429

See more details on using hashes here.

File details

Details for the file canonicalwebteam_discourse-7.5.0-py3-none-any.whl.

File metadata

File hashes

Hashes for canonicalwebteam_discourse-7.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 1e803b25d072eddc0a24ac81f44d224da2edfab0f4a47ecd3494456ec3b496b7
MD5 8abc63027d6d95c0938bdeaed8d7f272
BLAKE2b-256 75db5f59fbfeb0cd918399844cb9a099537f87d46cef2906fadc14e39a6c9ed7

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page