Skip to content
This repository has been archived by the owner on Nov 13, 2022. It is now read-only.

Escape Liquid Tags Present in Code / Code Output #66

Closed
drscotthawley opened this issue Feb 18, 2020 · 10 comments · Fixed by #70
Closed

Escape Liquid Tags Present in Code / Code Output #66

drscotthawley opened this issue Feb 18, 2020 · 10 comments · Fixed by #70
Assignees
Labels
bug Something isn't working enhancement New feature or request

Comments

@drscotthawley
Copy link
Contributor

drscotthawley commented Feb 18, 2020

I realize fastpages currently is only designed to work with Altair. Altair doesn't support 3D graphs, but plot.ly does. One of my notebooks uses plot.ly for interactive 3D graphs.

With a one-line change to nbdev (https://github.com/drscotthawley/nbdev/blob/master/nbdev/export2html.py), the embedded plot.ly 3D interactive graphs work fine, and the resulting html from nbdev/export2html shows these plots great. (Example HTML output of notebook from nbdev/export2html: https://hedges.belmont.edu/~shawley/2019-12-21-PCA-From-Scratch.html)

The problem is that further down in the fastpages build, the 'Rendering Liquid' step tries to treat some of the double-curly-brace javascript generated for plot.ly as if it were Liquid tags:

  Rendering Liquid: _posts/2019-12-21-PCA-From-Scratch.md
  Liquid Exception: Liquid syntax error (line 113): Variable '{{ x.observe(notebookContainer, {childList: true}' was not properly terminated with regexp: /\}\}/ in /github/workspace/_posts/2019-12-21-PCA-From-Scratch.md

The javascript code it's complaining about looks like this:

   if (notebookContainer) {{
       x.observe(notebookContainer, {childList: true});
   }}

I have no experience with Liquid, at all. My old Jekyll/Pages blog doesn't use it, so there was no conflict. But FastPages does use it. Is there any way to 'tell Liquid' to leave the plot.ly javascript alone and not try to interpret it as a tag? (I've googled around trying to solve this, but so far, no dice.)

Or perhaps there's a way to disable Liquid for certain files but not for the whole fastpages blog?

Run log: https://github.com/drscotthawley/devblog2/runs/451929186?

@drscotthawley
Copy link
Contributor Author

drscotthawley commented Feb 18, 2020

..So I see that the "monkey patch" in _action_files/nb2post.py simply changes the output file extension from .html to .md, and then Liquid runs on this md file but doesn't know it's operating on Javascript.

Would it work to reverse the order in the build? : Run Liquid on the file with the .html extension, and then change the extension to .md afterward?
(I have no experience with Liquid.)
Or else would it work to just allow a user to disable Liquid for certain files, maybe via a front-matter YAML tag at the top of the notebook?

EDIT: looks like from this StackOverflow post one could disable Liquid via the {% raw %} construction. Maybe a simple utility like

import glob 

# escape javascript sequences from Liquid via "raw" tag
def js_escape_liquid(fname='_notebooks/*.ipynb', dest='_posts/'):
    for nb_name in glob.glob(fname):
        md_name = dest+os.path.basename(nb_name).replace('.ipynb','.md')
        if os.path.exists(md_name):
            fin = open(md_name, "rt")
            data = fin.read()
            if '<script ' in data:
                data = data.replace('<script ', '\n{% raw %}\n<script ')
                data = data.replace('/script>', '/script>\n{% endraw %}\n ')
                fin.close()

                fin = open(md_name, "wt")
                fin.write(data)
            fin.close()

js_escape_liquid()

...Sticking that code at the end of _action_files/nb2post.py allows the build to complete, and the plot.ly plots work, but it does break a couple other things: external images no longer load, and each Markdown heading now ends with a "paragraph symbol".

@hamelsmu
Copy link
Member

Yeah I knew this was going to happen at some point, going to have to escape code blocks. Will put this on the backlog

@hamelsmu hamelsmu added bug Something isn't working enhancement New feature or request labels Feb 18, 2020
@hamelsmu
Copy link
Member

hamelsmu commented Feb 18, 2020

@drscotthawley this looks like it has been fixed in the latest version of nbdev, the one line change you applied looks like it was on an old version of nbdev. Do you want to test it out and see how it goes now?

I think you have two separate issues that you need to break out separately.

@hamelsmu
Copy link
Member

Actually I know how to fix this. working on it now

@drscotthawley
Copy link
Contributor Author

drscotthawley commented Feb 18, 2020

Ok the issue with external images is totally fixed in the latest nbdev. And the issue with nbdev & plot.ly is fixed. But the Liquid problems, which are the subject of this issue, those are still in play.

That little 'escape' code snippet above temporarily got plotl.ly working for me, but now the updates to nbdev seem to have broken that. Or they've introduced a similar issue with an embedded draw.io object. I'm not sure, here's what I see:

Liquid Exception: Invalid syntax for include tag: style="cursor:pointer;max-width:100%;" onclick="(function(img){if(img.wnd!=null&amp;&amp;!img.wnd.closed){img.wnd.focus();}else{var r=function(evt){if(evt.data=='ready'&amp;&amp;evt.source==img.wnd){img.wnd.postMessage(decodeURIComponent(img.getAttribute('src')),'*');window.removeEventListener('message',r);}};window.addEventListener('message',r);img.wnd=window.open('<a href="https://www.draw.io/?client=1&amp;lightbox=1&amp;edit=_blank&#39;);}})(this">https://www.draw.io/?client=1&amp;lightbox=1&amp;edit=_blank&#39;);}})(this</a>);" 
...
...and on and on

See this run: https://github.com/drscotthawley/devblog2/runs/453635808?check_suite_focus=true

So, I think this issue might still be unresolved.

@drscotthawley
Copy link
Contributor Author

drscotthawley commented Feb 18, 2020

I found a tricky interaction between nbdev & the fastpages build.

nbdev is replacing "<img src=" HTML code with Liquid "{% img src" tags.

This might not be a problem ordinarily, but when running with Liquid in fastpages, it can't handle long "src" strings.

So here's a question for both @sgugger and @hamelsmu:
Would it be better to NOT have nbdev translate the HTML "img src" tags into Liquid, or would it be better to provide fastpage's Liquid run with...whatever it needs to handle long src strings?

The original HTML code in question:

<img src="" style="cursor:pointer;max-width:100%;" onclick="(function(img){if(img.wnd!=null&&!img.wnd.closed){img.wnd.focus();}else{var r=function(evt){if(evt.data=='ready'&&evt.source==img.wnd){img.wnd.postMessage(decodeURIComponent(img.getAttribute('src')),'*');window.removeEventListener('message',r);}};window.addEventListener('message',r);img.wnd=window.open('https://www.draw.io/?client=1&lightbox=1&edit=_blank');}})(this);"/>

This is getting converted by nbdev to

{% include image.html style="cursor:pointer;max-width:100%;" onclick="(function(img){if(img.wnd!=null&amp;&amp;!img.wnd.closed){img.wnd.focus();}else{var r=function(evt){if(evt.data=='ready'&amp;&amp;evt.source==img.wnd){img.wnd.postMessage(decodeURIComponent(img.getAttribute('src')),'*');window.removeEventListener('message',r);}};window.addEventListener('message',r);img.wnd=window.open('<a href="https://www.draw.io/?client=1&amp;lightbox=1&amp;edit=_blank&#39;);}})(this">https://www.draw.io/?client=1&amp;lightbox=1&amp;edit=_blank&#39;);}})(this</a>);" file="" %}

@hamelsmu
Copy link
Member

@drscotthawley do you have a code snippet you can give me to test out your plotly graph? I can add it to the demo notebook.

@hamelsmu hamelsmu self-assigned this Feb 18, 2020
@hamelsmu hamelsmu changed the title Enhancement: resolve Liquid's attempts to treat plot.ly javascript as tags Escape Liquid Tags Present in Code / Code Output Feb 18, 2020
@hamelsmu
Copy link
Member

Closed this in favor of #71

@drscotthawley
Copy link
Contributor Author

drscotthawley commented Feb 18, 2020

Yes! Just created one. Check this out:
https://github.com/drscotthawley/devblog2/blob/master/_notebooks/2020-02-18-fastpages_tests.ipynb

EDIT: GitHub doesn't show the plot.ly graph, but when you load up the notebook in jupyter, or if you have HTML created by exporting HTML from jupyter (or from Google Colab), the graph appears.

@hamelsmu
Copy link
Member

@drscotthawley lets take the conversation to the new issue, thanks so much for all your troubleshooting! You nudged me to figure out how to escape liquid in code blocks!

Love your participation in this journey with us ❤️

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
bug Something isn't working enhancement New feature or request
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants