mirror of
https://github.com/django-components/django-components.git
synced 2025-07-07 17:34:59 +00:00

Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
442 lines
15 KiB
Python
442 lines
15 KiB
Python
from typing import List
|
|
|
|
from django.test import override_settings
|
|
from playwright.async_api import Error, Page
|
|
|
|
from django_components import types
|
|
from tests.django_test_setup import setup_test_config
|
|
from tests.e2e.utils import TEST_SERVER_URL, with_playwright
|
|
from tests.testutils import BaseTestCase
|
|
|
|
setup_test_config(
|
|
components={"autodiscover": False},
|
|
extra_settings={
|
|
"ROOT_URLCONF": "tests.test_dependency_manager",
|
|
},
|
|
)
|
|
|
|
urlpatterns: List = []
|
|
|
|
|
|
class _BaseDepManagerTestCase(BaseTestCase):
|
|
async def _create_page_with_dep_manager(self) -> Page:
|
|
page = await self.browser.new_page()
|
|
|
|
# Load the JS library by opening a page with the script, and then running the script code
|
|
# E.g. `http://localhost:54017/static/django_components/django_components.min.js`
|
|
script_url = TEST_SERVER_URL + "/static/django_components/django_components.min.js"
|
|
await page.goto(script_url)
|
|
|
|
# The page's body is the script code. We load it by executing the code
|
|
await page.evaluate(
|
|
"""
|
|
() => {
|
|
eval(document.body.textContent);
|
|
}
|
|
"""
|
|
)
|
|
|
|
# Ensure the body is clear
|
|
await page.evaluate(
|
|
"""
|
|
() => {
|
|
document.body.innerHTML = '';
|
|
document.head.innerHTML = '';
|
|
}
|
|
"""
|
|
)
|
|
|
|
return page
|
|
|
|
|
|
@override_settings(STATIC_URL="static/")
|
|
class DependencyManagerTests(_BaseDepManagerTestCase):
|
|
@with_playwright
|
|
async def test_script_loads(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
# Check the exposed API
|
|
keys = sorted(await page.evaluate("Object.keys(Components)"))
|
|
self.assertEqual(keys, ["createComponentsManager", "manager", "unescapeJs"])
|
|
|
|
keys = await page.evaluate("Object.keys(Components.manager)")
|
|
self.assertEqual(
|
|
keys, ["callComponent", "registerComponent", "registerComponentData", "loadScript", "markScriptLoaded"]
|
|
)
|
|
|
|
await page.close()
|
|
|
|
|
|
# Tests for `manager.loadScript()` / `manager.markAsLoaded()`
|
|
@override_settings(STATIC_URL="static/")
|
|
class LoadScriptTests(_BaseDepManagerTestCase):
|
|
@with_playwright
|
|
async def test_load_js_scripts(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
# JS code that loads a few dependencies, capturing the HTML after each action
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const headBeforeFirstLoad = document.head.innerHTML;
|
|
|
|
// Adds a script the first time
|
|
manager.loadScript('js', "<script src='/one/two'></script>");
|
|
const bodyAfterFirstLoad = document.body.innerHTML;
|
|
|
|
// Does not add it the second time
|
|
manager.loadScript('js', "<script src='/one/two'></script>");
|
|
const bodyAfterSecondLoad = document.body.innerHTML;
|
|
|
|
// Adds different script
|
|
manager.loadScript('js', "<script src='/four/three'></script>");
|
|
const bodyAfterThirdLoad = document.body.innerHTML;
|
|
|
|
const headAfterThirdLoad = document.head.innerHTML;
|
|
|
|
return {
|
|
bodyAfterFirstLoad,
|
|
bodyAfterSecondLoad,
|
|
bodyAfterThirdLoad,
|
|
headBeforeFirstLoad,
|
|
headAfterThirdLoad,
|
|
};
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data["bodyAfterFirstLoad"], '<script src="/one/two"></script>')
|
|
self.assertEqual(data["bodyAfterSecondLoad"], '<script src="/one/two"></script>')
|
|
self.assertEqual(
|
|
data["bodyAfterThirdLoad"], '<script src="/one/two"></script><script src="/four/three"></script>'
|
|
)
|
|
|
|
self.assertEqual(data["headBeforeFirstLoad"], data["headAfterThirdLoad"])
|
|
self.assertEqual(data["headBeforeFirstLoad"], "")
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_load_css_scripts(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
# JS code that loads a few dependencies, capturing the HTML after each action
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const bodyBeforeFirstLoad = document.body.innerHTML;
|
|
|
|
// Adds a script the first time
|
|
manager.loadScript('css', "<link href='/one/two'>");
|
|
const headAfterFirstLoad = document.head.innerHTML;
|
|
|
|
// Does not add it the second time
|
|
manager.loadScript('css', "<link herf='/one/two'>");
|
|
const headAfterSecondLoad = document.head.innerHTML;
|
|
|
|
// Adds different script
|
|
manager.loadScript('css', "<link href='/four/three'>");
|
|
const headAfterThirdLoad = document.head.innerHTML;
|
|
|
|
const bodyAfterThirdLoad = document.body.innerHTML;
|
|
|
|
return {
|
|
headAfterFirstLoad,
|
|
headAfterSecondLoad,
|
|
headAfterThirdLoad,
|
|
bodyBeforeFirstLoad,
|
|
bodyAfterThirdLoad,
|
|
};
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data["headAfterFirstLoad"], '<link href="/one/two">')
|
|
self.assertEqual(data["headAfterSecondLoad"], '<link href="/one/two">')
|
|
self.assertEqual(data["headAfterThirdLoad"], '<link href="/one/two"><link href="/four/three">')
|
|
|
|
self.assertEqual(data["bodyBeforeFirstLoad"], data["bodyAfterThirdLoad"])
|
|
self.assertEqual(data["bodyBeforeFirstLoad"], "")
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_does_not_load_script_if_marked_as_loaded(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
# JS code that loads a few dependencies, capturing the HTML after each action
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
// Adds a script the first time
|
|
manager.markScriptLoaded('css', '/one/two');
|
|
manager.markScriptLoaded('js', '/one/three');
|
|
|
|
manager.loadScript('css', "<link href='/one/two'>");
|
|
const headAfterFirstLoad = document.head.innerHTML;
|
|
|
|
manager.loadScript('js', "<script src='/one/three'></script>");
|
|
const bodyAfterSecondLoad = document.body.innerHTML;
|
|
|
|
return {
|
|
headAfterFirstLoad,
|
|
bodyAfterSecondLoad,
|
|
};
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data["headAfterFirstLoad"], "")
|
|
self.assertEqual(data["bodyAfterSecondLoad"], "")
|
|
|
|
await page.close()
|
|
|
|
|
|
# Tests for `manager.registerComponent()` / `registerComponentData()` / `callComponent()`
|
|
@override_settings(STATIC_URL="static/")
|
|
class CallComponentTests(_BaseDepManagerTestCase):
|
|
@with_playwright
|
|
async def test_calls_component_successfully(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
// Pretend that this HTML belongs to our component
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
let captured = null;
|
|
manager.registerComponent(compName, (data, ctx) => {
|
|
captured = { ctx, data };
|
|
return 123;
|
|
});
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
const result = manager.callComponent(compName, compId, inputHash);
|
|
|
|
// Serialize the HTML elements
|
|
captured.ctx.els = captured.ctx.els.map((el) => el.outerHTML);
|
|
|
|
return {
|
|
result,
|
|
captured,
|
|
};
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data["result"], 123)
|
|
self.assertEqual(
|
|
data["captured"],
|
|
{
|
|
"data": {
|
|
"hello": "world",
|
|
},
|
|
"ctx": {
|
|
"els": ['<div data-comp-id-12345=""> abc </div>'],
|
|
"id": "12345",
|
|
"name": "my_comp",
|
|
},
|
|
},
|
|
)
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_calls_component_successfully_async(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
// Pretend that this HTML belongs to our component
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
manager.registerComponent(compName, (data, ctx) => {
|
|
return Promise.resolve(123);
|
|
});
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
// Should be Promise
|
|
const result = manager.callComponent(compName, compId, inputHash);
|
|
const isPromise = `${result}` === '[object Promise]';
|
|
|
|
// Wrap the whole response in Promise, so we can add extra fields
|
|
return Promise.resolve(result).then((res) => ({
|
|
result: res,
|
|
isPromise,
|
|
}));
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data["result"], 123)
|
|
self.assertEqual(data["isPromise"], True)
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_error_in_component_call_do_not_propagate_sync(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
// Pretend that this HTML belongs to our component
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
manager.registerComponent(compName, (data, ctx) => {
|
|
throw Error('Oops!');
|
|
return 123;
|
|
});
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
const result = manager.callComponent(compName, compId, inputHash);
|
|
|
|
return result;
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(data, None)
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_error_in_component_call_do_not_propagate_async(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
// Pretend that this HTML belongs to our component
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
manager.registerComponent(compName, async (data, ctx) => {
|
|
throw Error('Oops!');
|
|
return 123;
|
|
});
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
const result = manager.callComponent(compName, compId, inputHash);
|
|
return Promise.allSettled([result]);
|
|
}"""
|
|
|
|
data = await page.evaluate(test_js)
|
|
|
|
self.assertEqual(len(data), 1)
|
|
self.assertEqual(data[0]["status"], "rejected")
|
|
self.assertIsInstance(data[0]["reason"], Error)
|
|
self.assertEqual(data[0]["reason"].message, "Oops!")
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_raises_if_component_element_not_in_dom(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
manager.registerComponent(compName, (data, ctx) => {
|
|
return 123;
|
|
});
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
// Should raise Error
|
|
manager.callComponent(compName, compId, inputHash);
|
|
}"""
|
|
|
|
with self.assertRaisesMessage(
|
|
Error, "Error: [Components] 'my_comp': No elements with component ID '12345' found"
|
|
):
|
|
await page.evaluate(test_js)
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_raises_if_input_hash_not_registered(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
manager.registerComponent(compName, (data, ctx) => {
|
|
return Promise.resolve(123);
|
|
});
|
|
|
|
// Should raise Error
|
|
manager.callComponent(compName, compId, inputHash);
|
|
}"""
|
|
|
|
with self.assertRaisesMessage(Error, "Error: [Components] 'my_comp': Cannot find input for hash 'input-abc'"):
|
|
await page.evaluate(test_js)
|
|
|
|
await page.close()
|
|
|
|
@with_playwright
|
|
async def test_raises_if_component_not_registered(self):
|
|
page = await self._create_page_with_dep_manager()
|
|
|
|
test_js: types.js = """() => {
|
|
const manager = Components.createComponentsManager();
|
|
|
|
const compName = 'my_comp';
|
|
const compId = '12345';
|
|
const inputHash = 'input-abc';
|
|
|
|
document.body.insertAdjacentHTML('beforeend', '<div data-comp-id-12345> abc </div>');
|
|
|
|
manager.registerComponentData(compName, inputHash, () => {
|
|
return { hello: 'world' };
|
|
});
|
|
|
|
// Should raise Error
|
|
manager.callComponent(compName, compId, inputHash);
|
|
}"""
|
|
|
|
with self.assertRaisesMessage(Error, "Error: [Components] 'my_comp': No component registered for that name"):
|
|
await page.evaluate(test_js)
|
|
|
|
await page.close()
|