from timApp.auth.accesstype import AccessType
from timApp.tests.server.timroutetest import TimRouteTest
from timApp.timdb.sqa import db
from timApp.user.user import User
[docs]class InlinePluginTest(TimRouteTest):
[docs] def test_inline_plugins(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali id=Lm7y6R7n5XIb}
*Hello* {#t1#}, {#tx:nonexistent#} and {#t2#} $x$
#- {defaultplugin=pali math_type=svg id=spOMcE20X2aX}
Hi {#t3#} $x$
#- {defaultplugin=pali id=Se0s8FDLbhOp}
{#t4 header: hi, footer: ho#}
"""
)
r = self.get(d.url, as_tree=True)
e = r.cssselect(".par")[0]
expected_json = self.create_plugin_json(
d,
"t1",
"Lm7y6R7n5XIb",
)
expected_json2 = self.create_plugin_json(
d,
"t2",
"Lm7y6R7n5XIb",
)
self.assert_same_html(
e,
rf"""
<div class="par" id="Lm7y6R7n5XIb" t="MHgyN2U5NDhhMA==" attrs='{{"defaultplugin": "pali"}}'>
<div tabindex="0" class="parContent">
<p>
<em>Hello</em> <tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t1"><span id="{d.id}.t1.Lm7y6R7n5XIb" data-plugin="/pali">
<pali-runner json="{self.make_base64(
expected_json)}"></pali-runner></span></tim-plugin-loader>, <span class="error" ng-non-bindable>Plugin nonexistent error: Plugin does not exist.</span> and <tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t2"><span id="{d.id}.t2.Lm7y6R7n5XIb" data-plugin="/pali"><pali-runner json="{self.make_base64(
expected_json2)}"></pali-runner></span></tim-plugin-loader>
<span class="math inline">\(x\)</span>
</p>
</div>
<div class="editline" tabindex="0" title="Click to edit this paragraph"></div>
<div class="readline" title="Click to mark this paragraph as read"></div>
</div>
""",
)
a = self.post_answer_no_abdata(
plugin_type="pali",
task_id=f"{d.id}.t2",
user_input={"userword": "aaaaaa"},
)
aid = a["savedNew"]
self.assertEqual(
{"savedNew": aid, "valid": True, "web": {"result": "saved"}}, a
)
self.assertIsInstance(aid, int)
r = self.get(d.url, as_tree=True)
e = r.cssselect(".par")[0]
s = {"userword": "aaaaaa"}
expected_json2 = self.create_plugin_json(
d,
"t2",
"Lm7y6R7n5XIb",
info={
"earlier_answers": 1,
"look_answer": False,
"max_answers": None,
"user_id": "testuser1",
"valid": True,
},
state=s,
)
self.assert_same_html(
e,
rf"""
<div class="par" id="Lm7y6R7n5XIb" t="MHgyN2U5NDhhMA==" attrs='{{"defaultplugin": "pali"}}'>
<div tabindex="0" class="parContent">
<p>
<em>Hello</em>
<tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t1">
<span id="{d.id}.t1.Lm7y6R7n5XIb" data-plugin="/pali">
<pali-runner json="{self.make_base64(expected_json)}"></pali-runner></span></tim-plugin-loader>,
<span class="error" ng-non-bindable>Plugin nonexistent error: Plugin does not exist.</span> and
<tim-plugin-loader type="full" answer-id="{aid}" class="pluginpali inlineplugin" task-id="{d.id}.t2">
<span id="{d.id}.t2.Lm7y6R7n5XIb" data-plugin="/pali"><pali-runner json="{self.make_base64(
expected_json2)}"></pali-runner></span></tim-plugin-loader>
<span class="math inline">\(x\)</span>
</p>
</div>
<div class="editline" tabindex="0" title="Click to edit this paragraph"></div>
<div class="readline" title="Click to mark this paragraph as read"></div>
</div>
""",
)
expected_json = self.create_plugin_json(
d,
"t3",
"spOMcE20X2aX",
)
self.assert_same_html(
r.cssselect(".par")[1],
f"""
<div class="par" id="spOMcE20X2aX" t="LTB4MTk4ZmYxOTQ=" attrs='{{"defaultplugin": "pali", "math_type": "svg"}}'>
<div tabindex="0" class="parContent">
<p>
Hi
<tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t3">
<span id="{d.id}.t3.spOMcE20X2aX" data-plugin="/pali">
<pali-runner json="{self.make_base64(expected_json)}"></pali-runner>
</span>
</tim-plugin-loader>
<span class="mathp inline"><img style="width:0.80327em; vertical-align:-0.06000em" src="" title="x"></span>
</p>
</div>
<div class="editline" tabindex="0" title="Click to edit this paragraph"></div>
<div class="readline" title="Click to mark this paragraph as read"></div>
</div>
""",
)
expected_json = self.create_plugin_json(
d,
"t4",
"Se0s8FDLbhOp",
markup={
"header": "hi",
"footer": "ho",
},
)
self.assert_same_html(
r.cssselect(".par")[2],
f"""
<div class="par" id="Se0s8FDLbhOp" t="LTB4NGU3YzFkYWM=" attrs='{{"defaultplugin": "pali"}}'>
<div tabindex="0" class="parContent">
<p>
<tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t4">
<span id="{d.id}.t4.Se0s8FDLbhOp" data-plugin="/pali">
<pali-runner json="{self.make_base64(expected_json)}"></pali-runner>
</span>
</tim-plugin-loader>
</p>
</div>
<div class="editline" tabindex="0" title="Click to edit this paragraph"></div>
<div class="readline" title="Click to mark this paragraph as read"></div>
</div>
""",
)
a = self.post_answer_no_abdata(
plugin_type="pali",
task_id=f"{d.id}.t5",
user_input={"userword": "aaaaaa"},
expect_status=400,
expect_content="Task not found in the document: t5",
)
[docs] def test_inline_plugin_no_html_escape(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali id=a3Xuyg1PF1l1}
{#t5 initword: #}
"""
)
r = self.get(d.url, as_tree=True)
# Make sure Dumbo won't escape plugin's error HTML.
self.assert_same_html(
r.cssselect(".parContent")[0],
f"""
<div tabindex="0" class="parContent">
<p>
<tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.t5">
<span id="{d.id}.t5.a3Xuyg1PF1l1" data-plugin="/pali">
<div class="pluginError">
The following fields have invalid values:
<ul>
<li>
initword: Field may not be null.
</li>
</ul>
</div>
</span>
</tim-plugin-loader>
</p>
</div>
""",
)
[docs] def test_inline_plugin_sanitize(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali}
<script>alert('hello')</script>
"""
)
r = self.get(d.url, as_tree=True)
self.assertFalse(r.cssselect(".parContent script"))
[docs] def test_multiline_inlineplugin(self):
"""Multiline markup in inlineplugins works."""
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali id=SSYigUyqdb7p}
{#t
initword: hi
#}
"""
)
r = self.get(d.url, as_tree=True)
e = r.cssselect(
f'.parContent > p > tim-plugin-loader[task-id="{d.id}.t"] > span > pali-runner'
)
self.assertTrue(e)
self.assert_plugin_json(
e[0],
self.create_plugin_json(
d,
"t",
par_id="SSYigUyqdb7p",
markup={"initword": "hi"},
),
)
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali id=SSYigUyqdb7p}
{#t
initword: hi
inputplaceholder: test
#}
"""
)
r = self.get(d.url, as_tree=True)
e = r.cssselect(
f'.parContent > p > tim-plugin-loader[task-id="{d.id}.t"] > span > pali-runner'
)
self.assertTrue(e)
self.assert_plugin_json(
e[0],
self.create_plugin_json(
d,
"t",
par_id="SSYigUyqdb7p",
markup={
"initword": "hi",
"inputplaceholder": "test",
},
),
)
[docs] def test_inline_plugin_error_html_no_p(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin="pali" id=a3Xuyg1PF1l1}
a {#x initword: #} b
"""
)
r = self.get(d.url, as_tree=True)
self.assert_same_html(
r.cssselect(".parContent")[0],
f"""
<div tabindex="0" class="parContent">
<p>
a
<tim-plugin-loader type="full" answer-id="" class="pluginpali inlineplugin" task-id="{d.id}.x">
<span id="{d.id}.x.a3Xuyg1PF1l1" data-plugin="/pali">
<div class="pluginError">
The following fields have invalid values:
<ul>
<li>
initword: Field may not be null.
</li>
</ul>
</div>
</span>
</tim-plugin-loader>
b
</p>
</div>""",
)
[docs] def test_inline_plugin_ref(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali id=SSYigUyqdb7p}
{#t}
"""
)
d2 = self.create_doc()
d2.document.add_paragraph_obj(
d.document.get_paragraphs()[0].create_reference(d2.document)
)
self.get(d2.url)
[docs] def test_inline_plugin_login(self):
self.login_test1()
d = self.create_doc(
initial_par="""
#- {defaultplugin=pali}
{#t#}
"""
)
u = d.url
User.get_anon().grant_access(d, AccessType.view)
db.session.commit()
self.logout()
r = self.get(u, as_tree=True).cssselect(".parContent tim-login-menu")
self.assertTrue(r)