2024-01-11 03:40:07 +00:00
|
|
|
"""Main entry point for the documentation build process."""
|
2023-04-22 12:40:29 +00:00
|
|
|
|
2024-08-09 23:26:03 +00:00
|
|
|
import json
|
2023-04-22 12:40:29 +00:00
|
|
|
import os
|
2024-05-26 12:09:08 +00:00
|
|
|
import subprocess
|
2024-08-01 04:58:26 +00:00
|
|
|
import textwrap
|
2023-04-22 12:40:29 +00:00
|
|
|
|
2024-05-26 12:09:08 +00:00
|
|
|
import requests
|
|
|
|
import yaml
|
|
|
|
|
2024-08-09 23:26:03 +00:00
|
|
|
# Cached settings dict values
|
|
|
|
global GLOBAL_SETTINGS
|
|
|
|
global USER_SETTINGS
|
|
|
|
|
|
|
|
# Read in the InvenTree settings file
|
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
settings_file = os.path.join(here, 'inventree_settings.json')
|
|
|
|
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(settings_file, encoding='utf-8') as sf:
|
2024-08-09 23:26:03 +00:00
|
|
|
settings = json.load(sf)
|
|
|
|
|
|
|
|
GLOBAL_SETTINGS = settings['global']
|
|
|
|
USER_SETTINGS = settings['user']
|
|
|
|
|
2024-05-26 12:09:08 +00:00
|
|
|
|
|
|
|
def get_repo_url(raw=False):
|
|
|
|
"""Return the repository URL for the current project."""
|
|
|
|
mkdocs_yml = os.path.join(os.path.dirname(__file__), 'mkdocs.yml')
|
|
|
|
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(mkdocs_yml, encoding='utf-8') as f:
|
2024-05-26 12:09:08 +00:00
|
|
|
mkdocs_config = yaml.safe_load(f)
|
|
|
|
repo_name = mkdocs_config['repo_name']
|
|
|
|
|
|
|
|
if raw:
|
|
|
|
return f'https://raw.githubusercontent.com/{repo_name}'
|
|
|
|
else:
|
|
|
|
return f'https://github.com/{repo_name}'
|
|
|
|
|
|
|
|
|
|
|
|
def check_link(url) -> bool:
|
|
|
|
"""Check that a provided URL is valid.
|
|
|
|
|
|
|
|
We allow a number attempts and a lengthy timeout,
|
|
|
|
as we do not want false negatives.
|
|
|
|
"""
|
|
|
|
CACHE_FILE = os.path.join(os.path.dirname(__file__), 'url_cache.txt')
|
|
|
|
|
|
|
|
# Keep a local cache file of URLs we have already checked
|
|
|
|
if os.path.exists(CACHE_FILE):
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(CACHE_FILE, encoding='utf-8') as f:
|
2024-05-26 12:09:08 +00:00
|
|
|
cache = f.read().splitlines()
|
|
|
|
|
|
|
|
if url in cache:
|
|
|
|
return True
|
|
|
|
|
|
|
|
attempts = 5
|
|
|
|
|
|
|
|
while attempts > 0:
|
|
|
|
response = requests.head(url, timeout=5000)
|
|
|
|
if response.status_code == 200:
|
|
|
|
# Update the cache file
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(CACHE_FILE, 'a', encoding='utf-8') as f:
|
2024-05-26 12:09:08 +00:00
|
|
|
f.write(f'{url}\n')
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
attempts -= 1
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
2023-04-22 12:40:29 +00:00
|
|
|
|
2024-07-30 02:28:36 +00:00
|
|
|
def get_build_enviroment() -> str:
|
|
|
|
"""Returns the branch we are currently building on, based on the environment variables of the various CI platforms."""
|
|
|
|
# Check if we are in ReadTheDocs
|
|
|
|
if os.environ.get('READTHEDOCS') == 'True':
|
|
|
|
return os.environ.get('READTHEDOCS_GIT_IDENTIFIER')
|
|
|
|
# We are in GitHub Actions
|
|
|
|
elif os.environ.get('GITHUB_ACTIONS') == 'true':
|
|
|
|
return os.environ.get('GITHUB_REF')
|
|
|
|
else:
|
|
|
|
return 'master'
|
|
|
|
|
|
|
|
|
2023-04-22 12:40:29 +00:00
|
|
|
def define_env(env):
|
2024-01-11 03:40:07 +00:00
|
|
|
"""Define custom environment variables for the documentation build process."""
|
2023-04-22 12:40:29 +00:00
|
|
|
|
2024-05-26 12:09:08 +00:00
|
|
|
@env.macro
|
2024-07-30 02:28:36 +00:00
|
|
|
def sourcedir(dirname, branch=None):
|
2024-05-26 12:09:08 +00:00
|
|
|
"""Return a link to a directory within the source code repository.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- dirname: The name of the directory to link to (relative to the top-level directory)
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
- A fully qualified URL to the source code directory on GitHub
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
- FileNotFoundError: If the directory does not exist, or the generated URL is invalid
|
|
|
|
"""
|
2024-07-30 02:28:36 +00:00
|
|
|
if branch == None:
|
|
|
|
branch = get_build_enviroment()
|
|
|
|
|
2024-05-26 12:09:08 +00:00
|
|
|
if dirname.startswith('/'):
|
|
|
|
dirname = dirname[1:]
|
|
|
|
|
|
|
|
# This file exists at ./docs/main.py, so any directory we link to must be relative to the top-level directory
|
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
root = os.path.abspath(os.path.join(here, '..'))
|
|
|
|
|
|
|
|
directory = os.path.join(root, dirname)
|
|
|
|
directory = os.path.abspath(directory)
|
|
|
|
|
|
|
|
if not os.path.exists(directory) or not os.path.isdir(directory):
|
|
|
|
raise FileNotFoundError(f'Source directory {dirname} does not exist.')
|
|
|
|
|
|
|
|
repo_url = get_repo_url()
|
|
|
|
|
|
|
|
url = f'{repo_url}/tree/{branch}/{dirname}'
|
|
|
|
|
|
|
|
# Check that the URL exists before returning it
|
|
|
|
if not check_link(url):
|
|
|
|
raise FileNotFoundError(f'URL {url} does not exist.')
|
|
|
|
|
|
|
|
return url
|
|
|
|
|
|
|
|
@env.macro
|
2024-07-30 02:28:36 +00:00
|
|
|
def sourcefile(filename, branch=None, raw=False):
|
2024-05-26 12:09:08 +00:00
|
|
|
"""Return a link to a file within the source code repository.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- filename: The name of the file to link to (relative to the top-level directory)
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
- A fully qualified URL to the source code file on GitHub
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
- FileNotFoundError: If the file does not exist, or the generated URL is invalid
|
|
|
|
"""
|
2024-07-30 02:28:36 +00:00
|
|
|
if branch == None:
|
|
|
|
branch = get_build_enviroment()
|
|
|
|
|
2024-05-26 12:09:08 +00:00
|
|
|
if filename.startswith('/'):
|
|
|
|
filename = filename[1:]
|
|
|
|
|
|
|
|
# This file exists at ./docs/main.py, so any file we link to must be relative to the top-level directory
|
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
root = os.path.abspath(os.path.join(here, '..'))
|
|
|
|
|
|
|
|
file_path = os.path.join(root, filename)
|
|
|
|
|
|
|
|
if not os.path.exists(file_path):
|
|
|
|
raise FileNotFoundError(f'Source file {filename} does not exist.')
|
|
|
|
|
|
|
|
repo_url = get_repo_url(raw=raw)
|
|
|
|
|
|
|
|
if raw:
|
|
|
|
url = f'{repo_url}/{branch}/{filename}'
|
|
|
|
else:
|
|
|
|
url = f'{repo_url}/blob/{branch}/{filename}'
|
|
|
|
|
|
|
|
# Check that the URL exists before returning it
|
|
|
|
if not check_link(url):
|
|
|
|
raise FileNotFoundError(f'URL {url} does not exist.')
|
|
|
|
|
|
|
|
return url
|
|
|
|
|
|
|
|
@env.macro
|
|
|
|
def invoke_commands():
|
|
|
|
"""Provides an output of the available commands."""
|
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
base = os.path.join(here, '..')
|
|
|
|
base = os.path.abspath(base)
|
|
|
|
tasks = os.path.join(base, 'tasks.py')
|
|
|
|
output = os.path.join(here, 'invoke-commands.txt')
|
|
|
|
|
|
|
|
command = f'invoke -f {tasks} --list > {output}'
|
|
|
|
|
|
|
|
assert subprocess.call(command, shell=True) == 0
|
|
|
|
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(output, encoding='utf-8') as f:
|
2024-05-26 12:09:08 +00:00
|
|
|
content = f.read()
|
|
|
|
|
|
|
|
return content
|
|
|
|
|
2023-04-22 12:40:29 +00:00
|
|
|
@env.macro
|
|
|
|
def listimages(subdir):
|
2024-01-11 03:40:07 +00:00
|
|
|
"""Return a listing of all asset files in the provided subdir."""
|
2023-04-22 12:40:29 +00:00
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
|
|
|
|
directory = os.path.join(here, 'docs', 'assets', 'images', subdir)
|
|
|
|
|
|
|
|
assets = []
|
|
|
|
|
2024-01-11 00:28:58 +00:00
|
|
|
allowed = ['.png', '.jpg']
|
2023-04-22 12:40:29 +00:00
|
|
|
|
|
|
|
for asset in os.listdir(directory):
|
2023-04-28 10:49:53 +00:00
|
|
|
if any(asset.endswith(x) for x in allowed):
|
2023-04-22 12:40:29 +00:00
|
|
|
assets.append(os.path.join(subdir, asset))
|
|
|
|
|
|
|
|
return assets
|
2024-05-22 00:17:01 +00:00
|
|
|
|
|
|
|
@env.macro
|
2024-08-19 22:41:06 +00:00
|
|
|
def includefile(filename: str, title: str, fmt: str = ''):
|
2024-05-27 12:25:23 +00:00
|
|
|
"""Include a file in the documentation, in a 'collapse' block.
|
2024-05-22 00:17:01 +00:00
|
|
|
|
2024-05-27 12:25:23 +00:00
|
|
|
Arguments:
|
|
|
|
- filename: The name of the file to include (relative to the top-level directory)
|
|
|
|
- title:
|
2024-08-19 22:41:06 +00:00
|
|
|
- fmt:
|
2024-05-27 12:25:23 +00:00
|
|
|
"""
|
|
|
|
here = os.path.dirname(__file__)
|
|
|
|
path = os.path.join(here, '..', filename)
|
|
|
|
path = os.path.abspath(path)
|
2024-05-22 00:17:01 +00:00
|
|
|
|
2024-05-27 12:25:23 +00:00
|
|
|
if not os.path.exists(path):
|
|
|
|
raise FileNotFoundError(f'Required file {path} does not exist.')
|
2024-05-22 00:17:01 +00:00
|
|
|
|
2024-08-19 22:26:32 +00:00
|
|
|
with open(path, encoding='utf-8') as f:
|
2024-05-22 00:17:01 +00:00
|
|
|
content = f.read()
|
|
|
|
|
2024-05-27 12:25:23 +00:00
|
|
|
data = f'??? abstract "{title}"\n\n'
|
2024-08-19 22:41:06 +00:00
|
|
|
data += f' ```{fmt}\n'
|
2024-05-22 00:17:01 +00:00
|
|
|
data += textwrap.indent(content, ' ')
|
|
|
|
data += '\n\n'
|
|
|
|
data += ' ```\n\n'
|
|
|
|
|
|
|
|
return data
|
2024-05-27 12:25:23 +00:00
|
|
|
|
|
|
|
@env.macro
|
|
|
|
def templatefile(filename):
|
|
|
|
"""Include code for a provided template file."""
|
|
|
|
base = os.path.basename(filename)
|
|
|
|
fn = os.path.join(
|
|
|
|
'src', 'backend', 'InvenTree', 'report', 'templates', filename
|
|
|
|
)
|
|
|
|
|
2024-08-19 22:41:06 +00:00
|
|
|
return includefile(fn, f'Template: {base}', fmt='html')
|
2024-08-09 23:26:03 +00:00
|
|
|
|
|
|
|
@env.macro
|
|
|
|
def rendersetting(setting: dict):
|
|
|
|
"""Render a provided setting object into a table row."""
|
|
|
|
name = setting['name']
|
|
|
|
description = setting['description']
|
2024-08-19 20:55:16 +00:00
|
|
|
default = setting.get('default')
|
|
|
|
units = setting.get('units')
|
2024-08-09 23:26:03 +00:00
|
|
|
|
|
|
|
return f'| {name} | {description} | {default if default is not None else ""} | {units if units is not None else ""} |'
|
|
|
|
|
|
|
|
@env.macro
|
|
|
|
def globalsetting(key: str):
|
|
|
|
"""Extract information on a particular global setting.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- key: The name of the global setting to extract information for.
|
|
|
|
"""
|
|
|
|
global GLOBAL_SETTINGS
|
|
|
|
setting = GLOBAL_SETTINGS[key]
|
|
|
|
|
|
|
|
return rendersetting(setting)
|
|
|
|
|
|
|
|
@env.macro
|
|
|
|
def usersetting(key: str):
|
|
|
|
"""Extract information on a particular user setting.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- key: The name of the user setting to extract information for.
|
|
|
|
"""
|
|
|
|
global USER_SETTINGS
|
|
|
|
setting = USER_SETTINGS[key]
|
|
|
|
|
|
|
|
return rendersetting(setting)
|