From 518a37f95de305a97d85cc6598de73f055ac097a Mon Sep 17 00:00:00 2001 From: "Karl O. Pinc" Date: Tue, 3 Oct 2023 22:00:30 -0500 Subject: [PATCH] Run comments taken from the docs through Sphinx to make the in-db text --- db/schemas/Makefile | 8 +++-- db/schemas/gen_comments.py | 45 +++++++++++++++++++++++----- doc/conf.py | 2 ++ doc/filter_refs.py | 61 ++++++++++++++++++++++++++++++++++++++ make_files/make_db.mk | 3 +- make_files/make_docs.mk | 8 ++++- 6 files changed, 116 insertions(+), 11 deletions(-) create mode 100755 doc/filter_refs.py diff --git a/db/schemas/Makefile b/db/schemas/Makefile index 7d4c34e..6020670 100644 --- a/db/schemas/Makefile +++ b/db/schemas/Makefile @@ -83,6 +83,10 @@ export DB_DEPENDS := $(patsubst %,../../%,$(DB_DEPENDS)) Makefile \ # _order_ in which they should be used to create db objects. SCHEMA_DIRS := tables triggers indexes functions views comments +# The directories used in comment generation. +COMMENT_DIRS := ../../doc/sphinx-doc/comment_src \ + ../../doc/sphinx-doc/comment_txt + # Derived variables # (These shouldn't need editing.) # (The use of $(ORDER) and filtering is to ensure the correct ordering.) @@ -527,7 +531,7 @@ $(TABLE_DROP_FILES): comment_on.sql comment_off.sql: build_comment_files # The directories where comment sql goes -$(COMMENT_SCHEMA_PATHS): +$(COMMENT_SCHEMA_PATHS) $(COMMENT_DIRS): mkdir -p $@ # Having the template directory as a prerequsite not only creates @@ -556,7 +560,7 @@ gen_func_comment_tmpl: $(PSQL_DEPENDS) gen_func_comment_tmpl.sql .PHONY: build_comment_files build_comment_files: $(PSQL_DEPENDS) \ gen_func_comment_tmpl $(COMMENT_TEMPLATE_PATHS) \ - $(COMMENT_SCHEMA_PATHS) + $(COMMENT_SCHEMA_PATHS) $(COMMENT_DIRS) ./gen_comments.py $(HTML_SRCDIR) $(ORDER) # Aggregate comments in schemas into 2 single files. ./create_sql.sh '$(COMMENT_ON_FILES)' > comment_on.sql diff --git a/db/schemas/gen_comments.py b/db/schemas/gen_comments.py index b52d1d4..ee8a347 100755 --- a/db/schemas/gen_comments.py +++ b/db/schemas/gen_comments.py @@ -59,6 +59,7 @@ Relies on the commented database objects' names being schema-unique. import pathlib import re +import subprocess import sys OUTPUT_DIR_FMT = "{schema}/comments" @@ -79,6 +80,26 @@ SQL = { "views": "COMMENT ON VIEW {name} IS ", } +RST_PATH = "../../doc/sphinx-doc/comment_src" +RST_FILE = f"{RST_PATH}/comment.rst" +TXT_PATH = "../../doc/sphinx-doc/comment_txt" +TXT_BASENAME = "comment" +TXT_FILE = f"{TXT_PATH}/{TXT_BASENAME}.txt" +SPHINX_CMD = [ + "../../doc/sphinx-venv/bin/sphinx-build", + "-q", + "-D", + f"root_doc={TXT_BASENAME}", + "-t", + "text_epilog", + "-c", + "../../doc", + "-b", + "text", + RST_PATH, + TXT_PATH, +] + class DuplicateError(Exception): """Db object exists in more than one schema""" @@ -211,14 +232,24 @@ def close_objs(db_objs): cf.shutdown() -def comment_obj(db_objs, obj_name, column, comment): +def rst_to_txt(rst): + """Return the textual representation of the supplied RST""" + # This is where wrapping all the functions below in an object + # would make it so we can keep state and not always open and close files. + with open(RST_FILE, "w") as rst_fd: + rst_fd.write(rst) + subprocess.run(SPHINX_CMD).check_returncode() + with open(TXT_FILE, "r") as txt_fd: + return txt_fd.read(None) + + +def comment_obj(db_obj, column, comment): """Write the comments for a database object""" - db_obj = db_objs[obj_name] if column: - sql = [SQL["column"].format(name=obj)] + sql = [SQL["column"].format(name=column)] else: sql = db_obj.sql - db_obj.cf.write(sql, "".join(comment).rstrip()) + db_obj.cf.write(sql, rst_to_txt("".join(comment).rstrip())) def comments_from_file(file, db_objs): @@ -231,14 +262,14 @@ def comments_from_file(file, db_objs): if line[0:indent] == prefix or line == "\n": comment.append(line[indent:]) else: - comment_obj(db_objs, obj_name, column, comment) + comment_obj(db_objs[obj_name], column, comment) in_match = False if not in_match: match = OBJ_SUMMARY.match(line) if match is not None: obj = match.group(2).lower() obj_name = obj.split(".")[0] - column = obj != obj_name + column = None if obj == obj_name else obj if obj_name in db_objs: indent = len(match.group(1)) prefix = " " * indent @@ -255,7 +286,7 @@ def comments_from_file(file, db_objs): ) lineno += 1 if in_match: - comment_obj(db_objs, obj_name, column, comment) + comment_obj(db_objs[obj_name], column, comment) def traverse_rst(rst_dir, db_objs): diff --git a/doc/conf.py b/doc/conf.py index 26a20aa..f7516a8 100644 --- a/doc/conf.py +++ b/doc/conf.py @@ -45,6 +45,8 @@ if tags.has('html_epilog'): epilog_file = 'html_epilog.inc' if tags.has('latex_epilog'): epilog_file = 'latex_epilog.inc' +if tags.has('text_epilog'): + epilog_file = 'text_epilog.inc' with open(pathlib.Path('sphinx-doc', epilog_file), 'r') as epilog: rst_epilog = epilog.read() diff --git a/doc/filter_refs.py b/doc/filter_refs.py new file mode 100755 index 0000000..a19d899 --- /dev/null +++ b/doc/filter_refs.py @@ -0,0 +1,61 @@ +#!/usr/bin/python3 +# Copyright (C) 2023 The Meme Factory, Inc. http://www.karlpinc.com/ +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +"""A filter. Remove :ref: constructs from replacement text""" +# Karl O. Pinc + +import sys +import re + +EXP = re.compile(r"(\s*..\s+\|[^\s]+\|\s+replace::\s+):ref:`(.*)\s+<.*>`") +BLOCK_START = re.compile(r"(\s*..\s+)") + + +def write_block(block): + match = EXP.match(block) + if match is None: + print(block, end="") + else: + print(f"{match.group(1)}{match.group(2)}") + + +def main(): + """Standard entrypoint""" + block = [] + in_block = False + for line in sys.stdin: + if in_block: + if line[0:indent] != prefix: + write_block(f'{"".join(block)}\n') + block = [] + in_block = False + else: + block.append(line[:-1]) + if not in_block: + match = BLOCK_START.match(line) + if match is None: + print(line, end="") + else: + indent = len(match.group(1)) + prefix = " " * indent + block.append(line[:-1]) + in_block = True + if in_block: + write_block(block) + + +if __name__ == "__main__": + main() diff --git a/make_files/make_db.mk b/make_files/make_db.mk index ae41b4f..aa1015c 100644 --- a/make_files/make_db.mk +++ b/make_files/make_db.mk @@ -440,7 +440,8 @@ db/schemas/createindexes.sql db/schemas/dropindexes.sql: $(PSQL_DEPENDS) # have to be changed to a rule that calls make on itself. .PHONY: db/schemas/comment_on.sql db/schemas/comment_off.sql db/schemas/comment_on.sql \ -db/schemas/comment_off.sql: $(HTML_RST_FILES) $(PSQL_DEPENDS) install_schemas +db/schemas/comment_off.sql: $(HTML_RST_FILES) $(PSQL_DEPENDS) \ + $(TEXT_EPILOG) install_schemas $(MAKE) -C db/schemas $(notdir $@) ## db/createtypes.sql Build file of sql which makes data types diff --git a/make_files/make_docs.mk b/make_files/make_docs.mk index a1648ed..3d2216f 100644 --- a/make_files/make_docs.mk +++ b/make_files/make_docs.mk @@ -158,6 +158,8 @@ LATEX_DIRS := $(sort $(dir $(LATEX_RST_FILES))) SOURCE_EPILOG := $(M4_BASEDIR)/epilog.inc.m4 HTML_EPILOG := $(SPHINX_DOCDIR)/html_epilog.inc LATEX_EPILOG := $(SPHINX_DOCDIR)/latex_epilog.inc +TEXT_EPILOG_IN := $(SPHINX_DOCDIR)/text_epilog.in +TEXT_EPILOG := $(SPHINX_DOCDIR)/text_epilog.inc # Where sphinx puts the pdf it builds SPHINX_PDF_A4_TARGET := $(SPHINX_BUILT_LATEX_A4)/$(PDF_NAME) @@ -404,12 +406,16 @@ $(HTML_EPILOG): $(SOURCE_EPILOG) $(DOC_DEPENDS) --define=sdb_output_fmt=$(SDB_HTML_FMT) \ $< \ > $@ -$(LATEX_EPILOG): $(SOURCE_EPILOG) $(DOC_DEPENDS) +$(LATEX_EPILOG) \ +$(TEXT_EPILOG_IN): $(SOURCE_EPILOG) $(DOC_DEPENDS) m4 $(M4_DOC_INCLUDE_ARGS) \ --define=sdb_output_fmt=not_html \ $< \ > $@ +$(TEXT_EPILOG): $(TEXT_EPILOG_IN) $(DOC_DEPENDS) $(DOCDIR)/filter_refs.py + ./$(DOCDIR)/filter_refs.py < $< > $@ + # Targets that make the views from their m4 files. # In theory, this would be useful for all database objects, but # we only _need_ the views for the docs so that's all we do. -- 2.34.1