Browse Source

Various change on PR 930

Alexandre Flament 3 years ago
parent
commit
1157462ff9

+ 40 - 52
searx/infopage/__init__.py

@@ -19,8 +19,9 @@ Usage in a Flask app route:
 
 
 """
 """
 
 
-__all__ = ['InfoPage', 'MistletoePage', 'InfoPageSet']
+__all__ = ['InfoPage', 'InfoPageSet']
 
 
+import os
 import os.path
 import os.path
 import logging
 import logging
 import typing
 import typing
@@ -33,16 +34,18 @@ import mistletoe
 from .. import get_setting
 from .. import get_setting
 from ..compat import cached_property
 from ..compat import cached_property
 from ..version import GIT_URL
 from ..version import GIT_URL
+from ..locales import LOCALE_NAMES
 
 
-logger = logging.getLogger('doc')
+
+logger = logging.getLogger('searx.infopage')
+_INFO_FOLDER = os.path.abspath(os.path.dirname(__file__))
 
 
 
 
 class InfoPage:
 class InfoPage:
     """A page of the :py:obj:`online documentation <InfoPageSet>`."""
     """A page of the :py:obj:`online documentation <InfoPageSet>`."""
 
 
-    def __init__(self, fname, base_url=None):
+    def __init__(self, fname):
         self.fname = fname
         self.fname = fname
-        self.base_url = base_url
 
 
     @cached_property
     @cached_property
     def raw_content(self):
     def raw_content(self):
@@ -66,19 +69,25 @@ class InfoPage:
                 t = l.strip('# ')
                 t = l.strip('# ')
         return t
         return t
 
 
+    @cached_property
+    def html(self):
+        """Render Markdown (CommonMark_) to HTML by using mistletoe_.
+
+        .. _CommonMark: https://commonmark.org/
+        .. _mistletoe: https://github.com/miyuchina/mistletoe
+
+        """
+        return mistletoe.markdown(self.content)
+
     def get_ctx(self):  # pylint: disable=no-self-use
     def get_ctx(self):  # pylint: disable=no-self-use
         """Jinja context to render :py:obj:`InfoPage.content`"""
         """Jinja context to render :py:obj:`InfoPage.content`"""
 
 
         def _md_link(name, url):
         def _md_link(name, url):
-            url = url_for(url)
-            if self.base_url:
-                url = self.base_url + url
+            url = url_for(url, _external=True)
             return "[%s](%s)" % (name, url)
             return "[%s](%s)" % (name, url)
 
 
         def _md_search(query):
         def _md_search(query):
-            url = '%s?q=%s' % (url_for('search'), urllib.parse.quote(query))
-            if self.base_url:
-                url = self.base_url + url
+            url = '%s?q=%s' % (url_for('search', _external=True), urllib.parse.quote(query))
             return '[%s](%s)' % (query, url)
             return '[%s](%s)' % (query, url)
 
 
         ctx = {}
         ctx = {}
@@ -89,33 +98,8 @@ class InfoPage:
 
 
         return ctx
         return ctx
 
 
-    def render(self):
-        """Render / return content"""
-        return self.content
-
-
-class MistletoePage(InfoPage):
-    """A HTML page of the :py:obj:`online documentation <InfoPageSet>`."""
-
-    def __init__(self, *args, **kwargs):
-        super().__init__(*args, **kwargs)
-
-    @cached_property
-    def html(self):
-        """HTML representation of this page"""
-        return self.render()
-
-    def render(self):
-        """Render Markdown (CommonMark_) to HTML by using mistletoe_.
-
-        .. _CommonMark: https://commonmark.org/
-        .. _mistletoe: https://github.com/miyuchina/mistletoe
-
-        """
-        return mistletoe.markdown(self.content)
-
-
-_INFO_FOLDER = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'info'))
+    def __repr__(self):
+        return f'<{self.__class__.__name__} fname={self.fname!r}>'
 
 
 
 
 class InfoPageSet:  # pylint: disable=too-few-public-methods
 class InfoPageSet:  # pylint: disable=too-few-public-methods
@@ -123,24 +107,26 @@ class InfoPageSet:  # pylint: disable=too-few-public-methods
 
 
     :param page_class: render online documentation by :py:obj:`InfoPage` parser.
     :param page_class: render online documentation by :py:obj:`InfoPage` parser.
     :type page_class: :py:obj:`InfoPage`
     :type page_class: :py:obj:`InfoPage`
+
+    :param info_folder: information directory
+    :type info_folder: str
     """
     """
 
 
-    def __init__(self, page_class: typing.Type[InfoPage], base_url=None):
-        self.page_class = page_class
-        self.base_url = base_url
-        self.CACHE: typing.Dict[tuple, InfoPage] = {}
+    def __init__(
+        self, page_class: typing.Optional[typing.Type[InfoPage]] = None, info_folder: typing.Optional[str] = None
+    ):
+        self.page_class = page_class or InfoPage
+        self.CACHE: typing.Dict[tuple, typing.Optional[InfoPage]] = {}
 
 
         # future: could be set from settings.xml
         # future: could be set from settings.xml
 
 
-        self.folder: str = _INFO_FOLDER
+        self.folder: str = info_folder or _INFO_FOLDER
         """location of the Markdwon files"""
         """location of the Markdwon files"""
 
 
-        self.i18n_origin: str = 'en'
+        self.locale_default: str = 'en'
         """default language"""
         """default language"""
 
 
-        self.l10n: typing.List = [
-            'en',
-        ]
+        self.locales: typing.List = [locale for locale in os.listdir(_INFO_FOLDER) if locale in LOCALE_NAMES]
         """list of supported languages (aka locales)"""
         """list of supported languages (aka locales)"""
 
 
         self.toc: typing.List = [
         self.toc: typing.List = [
@@ -160,12 +146,13 @@ class InfoPageSet:  # pylint: disable=too-few-public-methods
         :type locale: str
         :type locale: str
 
 
         """
         """
+        locale = locale or self.locale_default
+
         if pagename not in self.toc:
         if pagename not in self.toc:
             return None
             return None
-        if locale is not None and locale not in self.l10n:
+        if locale not in self.locales:
             return None
             return None
 
 
-        locale = locale or self.i18n_origin
         cache_key = (pagename, locale)
         cache_key = (pagename, locale)
         page = self.CACHE.get(cache_key)
         page = self.CACHE.get(cache_key)
 
 
@@ -176,16 +163,17 @@ class InfoPageSet:  # pylint: disable=too-few-public-methods
 
 
         fname = os.path.join(self.folder, locale, pagename) + '.md'
         fname = os.path.join(self.folder, locale, pagename) + '.md'
         if not os.path.exists(fname):
         if not os.path.exists(fname):
-            logger.error('file %s does not exists', fname)
+            logger.info('file %s does not exists', fname)
+            self.CACHE[cache_key] = None
             return None
             return None
 
 
-        page = self.page_class(fname, self.base_url)
+        page = self.page_class(fname)
         self.CACHE[cache_key] = page
         self.CACHE[cache_key] = page
         return page
         return page
 
 
     def all_pages(self, locale: typing.Optional[str] = None):
     def all_pages(self, locale: typing.Optional[str] = None):
-        """Iterate over all pages"""
-        locale = locale or self.i18n_origin
+        """Iterate over all pages of the TOC"""
+        locale = locale or self.locale_default
         for pagename in self.toc:
         for pagename in self.toc:
             page = self.get_page(pagename, locale)
             page = self.get_page(pagename, locale)
             yield pagename, page
             yield pagename, page

+ 0 - 0
searx/info/en/about.md → searx/infopage/en/about.md


+ 0 - 0
searx/info/en/search-syntax.md → searx/infopage/en/search-syntax.md


+ 3 - 3
searx/templates/oscar/info.html

@@ -2,9 +2,9 @@
 {% block title %}{{ active_page.title }} - {% endblock %}
 {% block title %}{{ active_page.title }} - {% endblock %}
 {% block content %}
 {% block content %}
 <ul class="nav nav-tabs">
 <ul class="nav nav-tabs">
-{% for pagename, page in all_pages('en') %}
-  <li {% if pagename == active_pagename %}class="active"{% endif %}>
-    <a href="{{pagename}}">{{page.title}}</a>
+  {% for pagename, page, locale in all_pages %}
+  <li>
+    <a href="{{ url_for('info', pagename=pagename, locale=locale) }}" {% if pagename == active_pagename %}class="active"{% endif %}>{{page.title}}</a>
   </li>
   </li>
 {% endfor %}
 {% endfor %}
 </ul>
 </ul>

+ 1 - 1
searx/templates/oscar/navbar.html

@@ -3,7 +3,7 @@
         <a href="{{ url_for('index') }}">{{ instance_name }}</a>{{- "" -}}
         <a href="{{ url_for('index') }}">{{ instance_name }}</a>{{- "" -}}
     </span>{{- "" -}}
     </span>{{- "" -}}
     <span class="{% if rtl %}pull-left{% else %}pull-right{% endif %}">{{- "" -}}
     <span class="{% if rtl %}pull-left{% else %}pull-right{% endif %}">{{- "" -}}
-        <a href="{{ url_for('info', pagename='about', locale=current_locale) }}">{{ _('about') }}</a>{{- "" -}}
+        <a href="{{ url_for('info', pagename='about') }}">{{ _('about') }}</a>{{- "" -}}
         <a href="{{ url_for('preferences') }}">{{ _('preferences') }}</a>{{- "" -}}
         <a href="{{ url_for('preferences') }}">{{ _('preferences') }}</a>{{- "" -}}
     </span>{{- "" -}}
     </span>{{- "" -}}
 </div>
 </div>

+ 1 - 1
searx/templates/simple/base.html

@@ -58,7 +58,7 @@
   </main>
   </main>
   <footer>
   <footer>
     <p>
     <p>
-    {{ _('Powered by') }} <a href="{{ url_for('info', pagename='about', locale=current_locale) }}">searxng</a> - {{ searx_version }} — {{ _('a privacy-respecting, hackable metasearch engine') }}<br/>
+    {{ _('Powered by') }} <a href="{{ url_for('info', pagename='about') }}">searxng</a> - {{ searx_version }} — {{ _('a privacy-respecting, hackable metasearch engine') }}<br/>
         <a href="{{ searx_git_url }}">{{ _('Source code') }}</a> |
         <a href="{{ searx_git_url }}">{{ _('Source code') }}</a> |
         <a href="{{ get_setting('brand.issue_url') }}">{{ _('Issue tracker') }}</a> |
         <a href="{{ get_setting('brand.issue_url') }}">{{ _('Issue tracker') }}</a> |
         <a href="{{ url_for('stats') }}">{{ _('Engine stats') }}</a> |
         <a href="{{ url_for('stats') }}">{{ _('Engine stats') }}</a> |

+ 2 - 2
searx/templates/simple/info.html

@@ -2,9 +2,9 @@
 {% block title %}{{ active_page.title }} - {% endblock %}
 {% block title %}{{ active_page.title }} - {% endblock %}
 {% block content %}
 {% block content %}
 <ul class="tabs">
 <ul class="tabs">
-{% for pagename, page in all_pages('en') %}
+{% for pagename, page, locale in all_pages %}
   <li>
   <li>
-    <a href="{{pagename}}" {% if pagename == active_pagename %}class="active"{% endif %}>{{page.title}}</a>
+    <a href="{{ url_for('info', pagename=pagename, locale=locale) }}" {% if pagename == active_pagename %}class="active"{% endif %}>{{page.title}}</a>
   </li>
   </li>
 {% endfor %}
 {% endfor %}
 </ul>
 </ul>

+ 17 - 5
searx/webapp.py

@@ -383,6 +383,11 @@ def url_for_theme(endpoint: str, override_theme: Optional[str] = None, **values)
         if file_hash:
         if file_hash:
             values['filename'] = filename_with_theme
             values['filename'] = filename_with_theme
             suffix = "?" + file_hash
             suffix = "?" + file_hash
+    if endpoint == 'info' and 'locale' not in values:
+        locale = request.preferences.get_value('locale')
+        if _INFO_PAGES.get_page(values['pagename'], locale) is None:
+            locale = _INFO_PAGES.locale_default
+        values['locale'] = locale
     return url_for(endpoint, **values) + suffix
     return url_for(endpoint, **values) + suffix
 
 
 
 
@@ -905,23 +910,30 @@ def about():
     return redirect(url_for('info', pagename='about', locale=locale))
     return redirect(url_for('info', pagename='about', locale=locale))
 
 
 
 
-_INFO_PAGES = infopage.InfoPageSet(infopage.MistletoePage)
+_INFO_PAGES = infopage.InfoPageSet()
 
 
 
 
 @app.route('/info/<locale>/<pagename>', methods=['GET'])
 @app.route('/info/<locale>/<pagename>', methods=['GET'])
 def info(pagename, locale):
 def info(pagename, locale):
     """Render page of online user documentation"""
     """Render page of online user documentation"""
 
 
-    locale = locale or request.preferences.get_value('locale')
     page = _INFO_PAGES.get_page(pagename, locale)
     page = _INFO_PAGES.get_page(pagename, locale)
-    if page is None:
-        page = _INFO_PAGES.get_page(pagename)
     if page is None:
     if page is None:
         flask.abort(404)
         flask.abort(404)
 
 
+    def all_pages():
+        user_locale = request.preferences.get_value('locale')
+        for for_pagename, for_page in _INFO_PAGES.all_pages(user_locale):
+            for_locale = locale
+            if for_page is None:
+                # we are sure that for_pagename != pagename
+                for_page = _INFO_PAGES.get_page(for_pagename, _INFO_PAGES.locale_default)
+                for_locale = _INFO_PAGES.locale_default
+            yield for_pagename, for_page, for_locale
+
     return render(
     return render(
         'info.html',
         'info.html',
-        all_pages=_INFO_PAGES.all_pages,
+        all_pages=all_pages(),
         active_page=page,
         active_page=page,
         active_pagename=pagename,
         active_pagename=pagename,
     )
     )

+ 18 - 21
searxng_extra/docs_prebuild

@@ -8,24 +8,26 @@
 import sys
 import sys
 import os.path
 import os.path
 import time
 import time
+from contextlib import contextmanager
 from searx import settings, get_setting
 from searx import settings, get_setting
 from searx.infopage import InfoPageSet, InfoPage
 from searx.infopage import InfoPageSet, InfoPage
 
 
+
 _doc_user = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'docs', 'user'))
 _doc_user = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', 'docs', 'user'))
 
 
-def main():
 
 
-    DOC = None
+def main():
     base_url = get_setting('server.base_url', None)
     base_url = get_setting('server.base_url', None)
-
     if base_url:
     if base_url:
-        DOC = _render_all_with_flask_ctx(base_url)
+        infopageset_ctx = _instance_infosetset_ctx(base_url)
     else:
     else:
-        DOC = _render_all()
-    for pagename, page in DOC.all_pages('en'):
-        fname = os.path.join(_doc_user, os.path.basename(page.fname))
-        with open(fname, 'w') as f:
-            f.write(page.content)
+        infopageset_ctx = _offline_infosetset_ctx()
+
+    with infopageset_ctx as infopageset:
+        for _, page in infopageset.all_pages('en'):
+            fname = os.path.join(_doc_user, os.path.basename(page.fname))
+            with open(fname, 'w') as f:
+                f.write(page.content)
 
 
 
 
 class OfflinePage(InfoPage):
 class OfflinePage(InfoPage):
@@ -41,21 +43,17 @@ class OfflinePage(InfoPage):
         return ctx
         return ctx
 
 
 
 
-def _render_all():
-    DOC = InfoPageSet(OfflinePage)
-    for pagename, page in DOC.all_pages('en'):
-        page.render()
-    return DOC
-
-
-def _render_all_with_flask_ctx(base_url):
+@contextmanager
+def _offline_infosetset_ctx():
+    yield InfoPageSet(OfflinePage)
 
 
-    DOC = InfoPageSet(InfoPage, base_url)
 
 
+@contextmanager
+def _instance_infosetset_ctx(base_url):
     # The url_for functions in the jinja templates need all routes to be
     # The url_for functions in the jinja templates need all routes to be
     # registered in the Flask app.
     # registered in the Flask app.
 
 
-    settings['server']['secret_key'] = "x"
+    settings['server']['secret_key'] = ''
     from searx.webapp import app
     from searx.webapp import app
 
 
     # Specify base_url so that url_for() works for base_urls.  If base_url is
     # Specify base_url so that url_for() works for base_urls.  If base_url is
@@ -63,8 +61,7 @@ def _render_all_with_flask_ctx(base_url):
     # generics (see flaskfix.py).
     # generics (see flaskfix.py).
 
 
     with app.test_request_context(base_url=base_url):
     with app.test_request_context(base_url=base_url):
-        for pagename, page in DOC.all_pages('en'):
-            page.render()
+        yield InfoPageSet()
 
 
     # The searx.webapp import from above fires some HTTP requests, thats
     # The searx.webapp import from above fires some HTTP requests, thats
     # why we get a RuntimeError::
     # why we get a RuntimeError::