-
Notifications
You must be signed in to change notification settings - Fork 456
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Template improvements #3760
base: master
Are you sure you want to change the base?
Template improvements #3760
Conversation
0f3c362
to
b46b121
Compare
docs/theming.rst
Outdated
@@ -130,8 +134,9 @@ The following keys are currently supported: | |||
|
|||
The parent is so you don’t have to create a full theme each time: just | |||
create an empty theme, set the parent, and add the bits you want modified. | |||
You **must** define a parent, otherwise many features won’t work due to | |||
missing templates, messages, and assets. | |||
It is strongly recommended you define a parent. If you don't, many features |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nope. Every theme must inherit from at least base
/base-jinja
, we explicitly refuse to support any other scenario (even if the code does not prevent this, we would prefer not to admit that it is a possibility, and we reserve the right to break it at any time.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have a humble nice little blog in mind. For that, I absolutely want my HTML generated without any <div>
in it. I want to prove that semantic HTML can be done. I also think that my little web site should work nicely without any Javascript (ROCA-style). This is not theory, this has really been my intention when I came to Nikola in the first place. If something in the templating changes, I'm willing to accept that my site will break on Nikola update and I'll have to fix.
I am willing to accept that I'll have to do a lot of theme customization to get that. As a platform as it stands now, Nikola would technically let me do this.
You are saying that Nikola strategically does not support my use case and I should go look elsewhere?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
See https://www.famsik.de/opinions for background. In my opinion, Nikola could serve nicely for people who want to do experiments with stuff like that. But accepting that as a valid Nikola use case is a project-political decision.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thinking about it a bit further: From my point of view, it would be useful if Nikola would simply officially state that
- it is possible to go without a parent template
- things might break on any Nikola version update, as new template files become required from the code.
As for support, that would mean for Nokia a new template requirement needs to be announced in the change file.
Would that be something you'd be willing to do if asked nicely?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here’s some example wording that might work:
While it is possible to create a theme without a parent, it is strongly discouraged and not officially supported. The
base
andbase-jinja
themes provide assets, messages, and generic templates that Nikola expects to be able to use in all sites. That said, if you are making something very custom, Nikola will not prevent the creation of a theme withoutbase
, but you will need to manually determine which templates and messages are required in your theme.
It’s unlikely we’ll ever introduce a new template, but I guess it might be mentioned in the changelog if that happens.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As far as the changelog is concerned: Anything in the templates can be user-changed. So I think Nikola has to mention any change in the template arena in the changelog anyway. I see that as independent of the question "with base or not".
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some stance like the one you suggest would answer my concerns. Thanks!
For some detail fine-tuning: I would prefer to warn people of consequences, yes, but otherwise treat them as adults who are able to make grown-up decisions. So I would think it nicer if we left out the "strongly discouraged" part. Maybe warn people that growing their own theme is harder than they might think at first sight.
As for the "not officially supported": Before people file a bug, we should require them to add the pertinent parent. If that makes the bug go away, it must not be filed.
Anything else the "not officially supported" entails?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
“Strongly discouraged” and “not officially supported” are meant to suggest “don’t do this unless you really know what you’re doing”. For user-facing software, phrases like this are a good idea IMO to clearly show the limitations.
“Not officially supported” means basically what you said, i.e. we won’t help with issues caused by a parentless theme, and we don’t guarantee it will always work.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Via the polyfill discussion, a solid reason surfaced why Nikola might rethink its policy of not officially supporting templates without parent.
Where GDPR rules, one needs a certain written agreement with any company before sending web traffic their way. This is a consequence of GDPR art 28, in particular section 2.. (I've personally been one of many developers who ported all references to Google fonts from external to locally hosted, in the web presence of a big Germany company (some 300.000+ employed).)
I believe small, private web sites are inside the target group of Nikola. People responsible for such web sites might not have the time and legal power to establish any such agreements legally needed. So they may choose to use no external resources whatsoever, but self-host everything their web site needs.
Nikola makes not promises regarding self-hosting presently. Indeed, build-in Nikola templates freely use external resources. Any Nikola version update might conceivably introduce or change external references in the HTML produced. Wanting to play it safe is a very solid reason to use a template without parent.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Generally there's the option use_cdn
, but as #3763 shows this isn't always respected, like for polyfill.io
.
I'm personally also using a template without a parent for my sites. (More precisely I have my own base template, and some templates deriving from my base.)
@@ -184,8 +196,16 @@ so ``post.tmpl`` only define the content, and the layout is inherited from ``bas | |||
|
|||
Another concept is theme inheritance. You do not need to duplicate all the | |||
default templates in your theme — you can just override the ones you want | |||
changed, and the rest will come from the parent theme. (Every theme needs a | |||
parent.) | |||
changed, and the rest will come from the parent theme. If your theme does not |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Again, unsupported and should not be documented.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you think about the compromise "we document that this is unsupported", which is what I'm trying to do here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’m still not a fan of documenting it at all, but I guess it can stay as-is.
# When this flag is set, fewer exceptions are handled internally; | ||
# instead they are left unhandled for the run time system to deal with them, | ||
# which typically leads to the stack traces being exposed. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
# When this flag is set, fewer exceptions are handled internally; | |
# instead they are left unhandled for the run time system to deal with them, | |
# which typically leads to the stack traces being exposed. | |
# A flag to show tracebacks of unhandled exceptions. | |
# An alternative to the DEBUG flag with less noise. |
(NIKOLA_DEBUG
used to be very spammy with yapsy, it’s still not great to have it always on; export NIKOLA_SHOW_TRACEBACKS=1
is in my .zshrc
though.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The wording "of unhandled exceptions" suggests whether the exceptions is handled or not is independent of this setting. As far as I remember, this is not the case.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Technically, all exceptions are handled. This setting affects exceptions that bubble up to the global handler and therefore crash the process.
if jinja2 is None: | ||
lookup = None | ||
else: | ||
lookup: Optional[jinja2.Environment] = None |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Consider using a string annotation:
if jinja2 is None: | |
lookup = None | |
else: | |
lookup: Optional[jinja2.Environment] = None | |
lookup: 'Optional[jinja2.Environment]' = None |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could be done and would technically work. But hides the concern and leaves the reader puzzling why the string has been used, as opposed to the plain Optional[jinja2.Environment]
. By putting it into the else
part, the concern is clear and explicit.
nikola/utils.py
Outdated
if isinstance(string_or_iterable, (str, bytes)): | ||
return string_or_iterable | ||
elif isinstance(string_or_iterable, Iterable): | ||
if isinstance(string_or_iterable, Iterable): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This won’t work, please undo:
>>> isinstance("a", typing.Iterable)
True
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point, thanks!
But what we had previously also doesn't work, as bytes
isn't a str
.
How about calling str( )
for both str
(where it is a no-op) and bytes
(where you can be lucky, so it might work)?
nikola/utils.py
Outdated
|
||
>>> smartjoin('; ', 'foo, bar') | ||
'foo, bar' | ||
>>> smartjoin('; ', ['foo', 'bar']) | ||
'foo; bar' | ||
>>> smartjoin(' to ', ['count', 42]) | ||
'count to 42' | ||
|
||
The treatment of bytes (calling str(string_or_iterable)) is somewhat dubious. Is this needed? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Might be worth checking if bytes
is ever passed here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes!
But there's a lot in this PR already, so some other day (and maybe some other person). I'm content with leaving the mess as it is, just improving the type hints.
b46b121
to
48ec10d
Compare
New features: - Trace template usage if env variable NIKOLA_TEMPLATES_TRACE is set. - Enable Jinja2 extensions via theme configuration. - Clarify template documentation: A parent template is not strictly needed. Additional code improvements helpful for those developing templates who actually read the code: - Added several typing hints and improved others. - A few code comments were added or improved.
…te engine. Also: Various type hint improvements.
Also some improvements to make the template handling code better and more readable; mostly type hints.
7395407
to
a319b4b
Compare
@@ -184,8 +196,16 @@ so ``post.tmpl`` only define the content, and the layout is inherited from ``bas | |||
|
|||
Another concept is theme inheritance. You do not need to duplicate all the | |||
default templates in your theme — you can just override the ones you want | |||
changed, and the rest will come from the parent theme. (Every theme needs a | |||
parent.) | |||
changed, and the rest will come from the parent theme. If your theme does not |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’m still not a fan of documenting it at all, but I guess it can stay as-is.
If you set the environment variable ``NIKOLA_TEMPLATES_TRACE`` to any non-empty value | ||
(``true`` is recommended), Nikola will log template usage, both on output and also | ||
into a file ``templates_log.txt``. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Unnecessary detail.
If you set the environment variable ``NIKOLA_TEMPLATES_TRACE`` to any non-empty value | |
(``true`` is recommended), Nikola will log template usage, both on output and also | |
into a file ``templates_log.txt``. | |
If you set the environment variable ``NIKOLA_TEMPLATES_TRACE`` to ``true``, | |
Nikola will log template usage, both on output and also into a file ``templates_log.txt``. |
# When this flag is set, fewer exceptions are handled internally; | ||
# instead they are left unhandled for the run time system to deal with them, | ||
# which typically leads to the stack traces being exposed. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Technically, all exceptions are handled. This setting affects exceptions that bubble up to the global handler and therefore crash the process.
@@ -61,7 +60,7 @@ class BasePlugin: | |||
"""Base plugin class.""" | |||
|
|||
logger = None | |||
site: 'nikola.nikola.Nikola' | |||
site: Optional['nikola.nikola.Nikola'] = None # NOQA # Fixme: Circular import |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not really something we can ever fix.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually, yes we can. And I have a feeling we should.
We could have an abstract base class, maybe Site
, for site
that has all the members that are actually used by whoever uses site
, and Nikola
could implement Site
. Then this member here could be of type Optional[Site]
and - puff - the need to import Nikola
has vanished.
I have a feeling this would actually be an improvement, not only because of this import problem. The site
is something we use in a lot of places, and separating contract (new Site
type) and implementation (Nikola
type) might be a good idea.
Would you like to have a new issue towards that end? I'm volunteering to write it.
"""Set the list of folders where templates are located and cache.""" | ||
raise NotImplementedError() | ||
|
||
def user_engine_factory(self, factory: Callable[..., Any]) -> None: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Consider a different name that includes a verb. Perhaps set_user_engine_factory
?
Pull Request Checklist
Description
New features:
conf.py
factory method.Additional code improvements
helpful for those developing templates who actually read the code:
Details
The commit message of the first commit can be used as the commit message of this entire PR.