Spaces:
Running
Running
feat: All htmx examples
Browse files- .vscode/settings.json +2 -2
- public/htmx_dialogs_custom.css +103 -0
- public/script.js +0 -1
- pyproject.toml +2 -2
- src/tutorial/__init__.py +8 -1
- src/tutorial/htmx/_14_file_upload_input.py +59 -0
- src/tutorial/htmx/_15_dialogs_browser.py +37 -0
- src/tutorial/htmx/_16_dialogs_uikit.py +90 -0
- src/tutorial/htmx/_17_dialogs_bootstrap.py +55 -0
- src/tutorial/htmx/_18_dialogs_custom.py +50 -0
- src/tutorial/htmx/_19_tabs_using_hateoas.py +53 -0
- src/tutorial/htmx/_20_tabs_using_javascript.py +59 -0
- src/tutorial/htmx/_21_keyboard_shortcuts.py +32 -0
- src/tutorial/htmx/_22_sortable.py +76 -0
- src/tutorial/htmx/_23_update_other_content.py +239 -0
- src/tutorial/htmx/_24_confirm_custom.py +82 -0
- src/tutorial/htmx/_25_async_auth.py +65 -0
- src/tutorial/htmx/_26_web_components.py +50 -0
- src/tutorial/htmx/_27_move_before.py +60 -0
- uv.lock +101 -232
.vscode/settings.json
CHANGED
@@ -3,8 +3,8 @@
|
|
3 |
"editor.formatOnSave": true,
|
4 |
"editor.defaultFormatter": "charliermarsh.ruff",
|
5 |
"editor.codeActionsOnSave": {
|
6 |
-
"source.fixAll": "explicit"
|
7 |
-
"source.organizeImports": "explicit"
|
8 |
}
|
9 |
},
|
10 |
"notebook.formatOnSave.enabled": true,
|
|
|
3 |
"editor.formatOnSave": true,
|
4 |
"editor.defaultFormatter": "charliermarsh.ruff",
|
5 |
"editor.codeActionsOnSave": {
|
6 |
+
"source.fixAll": "explicit"
|
7 |
+
// "source.organizeImports": "explicit"
|
8 |
}
|
9 |
},
|
10 |
"notebook.formatOnSave.enabled": true,
|
public/htmx_dialogs_custom.css
ADDED
@@ -0,0 +1,103 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
/***** MODAL DIALOG ****/
|
2 |
+
#modal {
|
3 |
+
/* Underlay covers entire screen. */
|
4 |
+
position: fixed;
|
5 |
+
top: 0px;
|
6 |
+
bottom: 0px;
|
7 |
+
left: 0px;
|
8 |
+
right: 0px;
|
9 |
+
background-color: rgba(0, 0, 0, 0.5);
|
10 |
+
z-index: 1000;
|
11 |
+
|
12 |
+
/* Flexbox centers the .modal-content vertically and horizontally */
|
13 |
+
display: flex;
|
14 |
+
flex-direction: column;
|
15 |
+
align-items: center;
|
16 |
+
|
17 |
+
/* Animate when opening */
|
18 |
+
animation-name: fadeIn;
|
19 |
+
animation-duration: 150ms;
|
20 |
+
animation-timing-function: ease;
|
21 |
+
}
|
22 |
+
|
23 |
+
#modal > .modal-underlay {
|
24 |
+
/* underlay takes up the entire viewport. This is only
|
25 |
+
required if you want to click to dismiss the popup */
|
26 |
+
position: absolute;
|
27 |
+
z-index: -1;
|
28 |
+
top: 0px;
|
29 |
+
bottom: 0px;
|
30 |
+
left: 0px;
|
31 |
+
right: 0px;
|
32 |
+
}
|
33 |
+
|
34 |
+
#modal > .modal-content {
|
35 |
+
/* Position visible dialog near the top of the window */
|
36 |
+
margin-top: 10vh;
|
37 |
+
|
38 |
+
/* Sizing for visible dialog */
|
39 |
+
width: 80%;
|
40 |
+
max-width: 600px;
|
41 |
+
|
42 |
+
/* Display properties for visible dialog*/
|
43 |
+
border: solid 1px #999;
|
44 |
+
border-radius: 8px;
|
45 |
+
box-shadow: 0px 0px 20px 0px rgba(0, 0, 0, 0.3);
|
46 |
+
background-color: white;
|
47 |
+
padding: 20px;
|
48 |
+
|
49 |
+
/* Animate when opening */
|
50 |
+
animation-name: zoomIn;
|
51 |
+
animation-duration: 150ms;
|
52 |
+
animation-timing-function: ease;
|
53 |
+
}
|
54 |
+
|
55 |
+
#modal.closing {
|
56 |
+
/* Animate when closing */
|
57 |
+
animation-name: fadeOut;
|
58 |
+
animation-duration: 150ms;
|
59 |
+
animation-timing-function: ease;
|
60 |
+
}
|
61 |
+
|
62 |
+
#modal.closing > .modal-content {
|
63 |
+
/* Animate when closing */
|
64 |
+
animation-name: zoomOut;
|
65 |
+
animation-duration: 150ms;
|
66 |
+
animation-timing-function: ease;
|
67 |
+
}
|
68 |
+
|
69 |
+
@keyframes fadeIn {
|
70 |
+
0% {
|
71 |
+
opacity: 0;
|
72 |
+
}
|
73 |
+
100% {
|
74 |
+
opacity: 1;
|
75 |
+
}
|
76 |
+
}
|
77 |
+
|
78 |
+
@keyframes fadeOut {
|
79 |
+
0% {
|
80 |
+
opacity: 1;
|
81 |
+
}
|
82 |
+
100% {
|
83 |
+
opacity: 0;
|
84 |
+
}
|
85 |
+
}
|
86 |
+
|
87 |
+
@keyframes zoomIn {
|
88 |
+
0% {
|
89 |
+
transform: scale(0.9);
|
90 |
+
}
|
91 |
+
100% {
|
92 |
+
transform: scale(1);
|
93 |
+
}
|
94 |
+
}
|
95 |
+
|
96 |
+
@keyframes zoomOut {
|
97 |
+
0% {
|
98 |
+
transform: scale(1);
|
99 |
+
}
|
100 |
+
100% {
|
101 |
+
transform: scale(0.9);
|
102 |
+
}
|
103 |
+
}
|
public/script.js
CHANGED
@@ -26,7 +26,6 @@ function init_main_page() {
|
|
26 |
window.document.addEventListener(
|
27 |
"SubappAfterRequest",
|
28 |
(e) => {
|
29 |
-
// console.log(e);
|
30 |
htmx.ajax("PUT", "/requests", {
|
31 |
target: "#request-list",
|
32 |
values: e.detail,
|
|
|
26 |
window.document.addEventListener(
|
27 |
"SubappAfterRequest",
|
28 |
(e) => {
|
|
|
29 |
htmx.ajax("PUT", "/requests", {
|
30 |
target: "#request-list",
|
31 |
values: e.detail,
|
pyproject.toml
CHANGED
@@ -22,12 +22,12 @@ dev-dependencies = [
|
|
22 |
"pytest-playwright>=0.5.2",
|
23 |
"pytest>=8.3.2",
|
24 |
"ruff>=0.6.3",
|
25 |
-
"devicorn",
|
26 |
]
|
27 |
|
28 |
[tool.uv.sources]
|
29 |
# fh-utils = { path = "../fh_utils", editable = true }
|
30 |
-
devicorn = { path = "../devicorn", editable = true }
|
31 |
|
32 |
[tool.ruff]
|
33 |
line-length = 120
|
|
|
22 |
"pytest-playwright>=0.5.2",
|
23 |
"pytest>=8.3.2",
|
24 |
"ruff>=0.6.3",
|
25 |
+
# "devicorn",
|
26 |
]
|
27 |
|
28 |
[tool.uv.sources]
|
29 |
# fh-utils = { path = "../fh_utils", editable = true }
|
30 |
+
# devicorn = { path = "../devicorn", editable = true }
|
31 |
|
32 |
[tool.ruff]
|
33 |
line-length = 120
|
src/tutorial/__init__.py
CHANGED
@@ -1,4 +1,5 @@
|
|
1 |
import importlib
|
|
|
2 |
from dataclasses import dataclass
|
3 |
from pathlib import Path
|
4 |
|
@@ -70,10 +71,16 @@ class RequestInfo:
|
|
70 |
|
71 |
@app.put("/requests")
|
72 |
def requests(r: RequestInfo):
|
|
|
|
|
|
|
|
|
|
|
73 |
return Div(**{"x-data": "{show: false}", "@click": "show = !show"})(
|
74 |
H4(x_text="(show?'▽':'▶') + ' " + r.verb.upper() + " " + r.path + "'"),
|
75 |
Div(**{"x-show": "show"})(
|
76 |
-
Div(Pre("Input: " + r.parameters)),
|
|
|
77 |
Div(Pre(r.response or "(empty response)"), style="max-height:150px;overflow:scroll;"),
|
78 |
),
|
79 |
)
|
|
|
1 |
import importlib
|
2 |
+
import json
|
3 |
from dataclasses import dataclass
|
4 |
from pathlib import Path
|
5 |
|
|
|
71 |
|
72 |
@app.put("/requests")
|
73 |
def requests(r: RequestInfo):
|
74 |
+
headers = json.loads(r.headers)
|
75 |
+
print(headers)
|
76 |
+
headers = {
|
77 |
+
k: v for k, v in headers.items() if k in ("HX-Trigger", "HX-Trigger-Name", "HX-Target", "HX-Prompt") and v
|
78 |
+
}
|
79 |
return Div(**{"x-data": "{show: false}", "@click": "show = !show"})(
|
80 |
H4(x_text="(show?'▽':'▶') + ' " + r.verb.upper() + " " + r.path + "'"),
|
81 |
Div(**{"x-show": "show"})(
|
82 |
+
Div(Pre("Input: " + r.parameters)) if r.parameters != "{}" else None,
|
83 |
+
Div(Pre("Headers: " + str(headers))) if headers else None,
|
84 |
Div(Pre(r.response or "(empty response)"), style="max-height:150px;overflow:scroll;"),
|
85 |
),
|
86 |
)
|
src/tutorial/htmx/_14_file_upload_input.py
ADDED
@@ -0,0 +1,59 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Button, Div, Form, Input, Label, fast_app
|
2 |
+
from starlette.datastructures import UploadFile
|
3 |
+
|
4 |
+
app, rt = fast_app()
|
5 |
+
|
6 |
+
form_name = "binaryForm"
|
7 |
+
|
8 |
+
|
9 |
+
def my_form(*children):
|
10 |
+
return Form(enctype="multipart/form-data", hx_swap="outerHTML", hx_post=submit, hx_target="this", id=form_name)(
|
11 |
+
Button("Submit"),
|
12 |
+
*children,
|
13 |
+
)
|
14 |
+
|
15 |
+
|
16 |
+
@app.get
|
17 |
+
def page():
|
18 |
+
return Div(
|
19 |
+
# Move file input out of the form
|
20 |
+
Label("An error should happen when you click on submit button. But the uploaded file should NOT be cleared"),
|
21 |
+
Input(form=form_name, type="file", name="binaryFile"),
|
22 |
+
my_form(),
|
23 |
+
)
|
24 |
+
|
25 |
+
|
26 |
+
@app.post
|
27 |
+
async def submit(binaryFile: UploadFile):
|
28 |
+
return my_form(Div("Error: Try again", style="color:red"))
|
29 |
+
|
30 |
+
|
31 |
+
@rt
|
32 |
+
def bad_form():
|
33 |
+
# This bad implementation
|
34 |
+
# Users are required to re-upload the file in case of error on other fields
|
35 |
+
return Form(hx_swap="outerHTML", hx_post=bad_form, hx_target="this")(
|
36 |
+
Input(type="file", name="binaryFile"),
|
37 |
+
Button("Submit"),
|
38 |
+
)
|
39 |
+
|
40 |
+
|
41 |
+
DESC = "Demonstrates how to preserve file inputs after form errors"
|
42 |
+
DOC = """
|
43 |
+
When using server-side error handling and validation with forms that include both primitive values and file inputs, the file input’s value is lost when the form returns with error messages. Consequently, users are required to re-upload the file, resulting in a less user-friendly experience.
|
44 |
+
|
45 |
+
To overcome the problem of losing file input value in simple cases, you can adopt the following approach:
|
46 |
+
|
47 |
+
Before:
|
48 |
+
|
49 |
+
::bad_form::
|
50 |
+
|
51 |
+
After:
|
52 |
+
|
53 |
+
::page submit my_form::
|
54 |
+
|
55 |
+
Form Restructuring: Move the binary file input outside the main form element in the HTML structure.
|
56 |
+
|
57 |
+
Using the form Attribute: Enhance the binary file input by adding the `form attribute` and setting its value to the ID of the main form. This linkage associates the binary file input with the form, even when it resides outside the form element.
|
58 |
+
"""
|
59 |
+
HEIGHT = "250px"
|
src/tutorial/htmx/_15_dialogs_browser.py
ADDED
@@ -0,0 +1,37 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Button, Div, fast_app
|
2 |
+
|
3 |
+
app, rt = fast_app()
|
4 |
+
|
5 |
+
|
6 |
+
@app.get
|
7 |
+
def page():
|
8 |
+
return Div()(
|
9 |
+
Button(
|
10 |
+
"Prompt Submission",
|
11 |
+
hx_post=submit,
|
12 |
+
hx_prompt="Enter a string",
|
13 |
+
hx_confirm="Are you sure?",
|
14 |
+
hx_target="#response",
|
15 |
+
),
|
16 |
+
Div(id="response"),
|
17 |
+
)
|
18 |
+
|
19 |
+
|
20 |
+
@app.post("/submit")
|
21 |
+
def submit(request, htmx: dict):
|
22 |
+
return f"User entered <i>{request.headers['HX-Prompt']}</i>"
|
23 |
+
|
24 |
+
|
25 |
+
DESC = "Demonstrates the prompt and confirm dialogs"
|
26 |
+
DOC = """
|
27 |
+
Dialogs can be triggered with the hx-prompt and hx-confirmattributes. These are triggered by the user interaction that would trigger the AJAX request, but the request is only sent if the dialog is accepted.
|
28 |
+
|
29 |
+
::page submit::
|
30 |
+
The value provided by the user to the prompt dialog is sent to the server in a HX-Prompt header. In this case, the server simply echos the user input back.
|
31 |
+
|
32 |
+
```
|
33 |
+
User entered <i>${response}</i>
|
34 |
+
```
|
35 |
+
"""
|
36 |
+
HEIGHT = "100px"
|
37 |
+
HTMX_URL = "https://htmx.org/examples/dialogs/"
|
src/tutorial/htmx/_16_dialogs_uikit.py
ADDED
@@ -0,0 +1,90 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import H2, Button, Div, Form, Input, Link, Script, fast_app
|
2 |
+
|
3 |
+
app, rt = fast_app(
|
4 |
+
hdrs=[
|
5 |
+
Script(src="https://unpkg.com/hyperscript.org"),
|
6 |
+
Link(rel="stylesheet", href="https://cdnjs.cloudflare.com/ajax/libs/uikit/3.5.9/css/uikit-core.min.css"),
|
7 |
+
],
|
8 |
+
pico=False,
|
9 |
+
)
|
10 |
+
|
11 |
+
|
12 |
+
@app.get
|
13 |
+
def page():
|
14 |
+
return Div()(
|
15 |
+
Button(
|
16 |
+
"Open Modal",
|
17 |
+
hx_get=modal,
|
18 |
+
hx_target="#modals-here",
|
19 |
+
_="on htmx:afterOnLoad wait 10ms then add .uk-open to #modal",
|
20 |
+
),
|
21 |
+
Div(id="modals-here"),
|
22 |
+
)
|
23 |
+
|
24 |
+
|
25 |
+
@app.get
|
26 |
+
def modal():
|
27 |
+
return Div(id="modal", cls="uk-modal", style="display:block;")(
|
28 |
+
Div(cls="uk-modal-dialog uk-modal-body")(
|
29 |
+
H2("Modal Dialog", cls="uk-modal-title"),
|
30 |
+
Form(
|
31 |
+
Div(cls="uk-margin")(Input(cls="uk-input", placeholder="What is Your Name?")),
|
32 |
+
Button(
|
33 |
+
"Save Changes",
|
34 |
+
cls="uk-button uk-button-primary",
|
35 |
+
type="button",
|
36 |
+
),
|
37 |
+
Button(
|
38 |
+
"Close",
|
39 |
+
id="cancelButton",
|
40 |
+
cls="uk-button uk-button-default",
|
41 |
+
type="button",
|
42 |
+
_="on click take .uk-open from #modal wait 200ms then remove #modal",
|
43 |
+
),
|
44 |
+
),
|
45 |
+
)
|
46 |
+
)
|
47 |
+
|
48 |
+
|
49 |
+
DESC = "Demonstrates modal dialogs using Bootstrap"
|
50 |
+
DOC = """
|
51 |
+
Many CSS toolkits include styles (and Javascript) for creating modal dialog boxes. This example shows how to use HTMX to display dynamic dialog using UIKit, and how to trigger its animation styles with little or no Javascript.
|
52 |
+
|
53 |
+
We start with a button that triggers the dialog, along with a DIV at the bottom of your markup where the dialog will be loaded:
|
54 |
+
|
55 |
+
This is an example of using HTMX to remotely load modal dialogs using UIKit. In this example we will use Hyperscript to demonstrate how cleanly that scripting language allows you to glue htmx and other libraries together.
|
56 |
+
|
57 |
+
::page::
|
58 |
+
|
59 |
+
This button uses a GET request to /modal when this button is clicked. The contents of this file will be added to the DOM underneath the #modals-here DIV.
|
60 |
+
|
61 |
+
Rather than using the standard UIKit Javascript library we are using a bit of Hyperscript, which triggers UIKit’s smooth animations. It is delayed by 10ms so that UIKit’s animations will run correctly.
|
62 |
+
|
63 |
+
Finally, the server responds with a slightly modified version of UIKit’s standard modal
|
64 |
+
|
65 |
+
::modal::
|
66 |
+
|
67 |
+
Hyperscript on the button and the form trigger animations when this dialog is completed or canceled. If you didn’t use this Hyperscript, the modals will still work but UIKit’s fade in animations will not be triggered.
|
68 |
+
|
69 |
+
You can, of course, use JavaScript rather than Hyperscript for this work, it’s just a lot more code:
|
70 |
+
|
71 |
+
```js
|
72 |
+
// This triggers the fade-in animation when a modal dialog is loaded and displayed
|
73 |
+
window.document.getElementById("showButton").addEventListener("htmx:afterOnLoad", function() {
|
74 |
+
setTimeout(function(){
|
75 |
+
window.document.getElementById("modal").classList.add("uk-open")
|
76 |
+
}, 10)
|
77 |
+
})
|
78 |
+
|
79 |
+
// This triggers the fade-out animation when the modal is closed.
|
80 |
+
window.document.getElementById("cancelButton").addEventListener("click", function() {
|
81 |
+
window.document.getElementById("modal").classList.remove("uk-open")
|
82 |
+
setTimeout(function(){
|
83 |
+
window.document.getElementById("modals-here").innerHTML = ""
|
84 |
+
,200
|
85 |
+
})
|
86 |
+
})
|
87 |
+
```
|
88 |
+
"""
|
89 |
+
HTMX_URL = "https://htmx.org/examples/modal-uikit/"
|
90 |
+
HEIGHT = "350px"
|
src/tutorial/htmx/_17_dialogs_bootstrap.py
ADDED
@@ -0,0 +1,55 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import H2, Button, Div, Link, Script, fast_app
|
2 |
+
|
3 |
+
app, rt = fast_app(
|
4 |
+
hdrs=[
|
5 |
+
Script(src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"),
|
6 |
+
Link(
|
7 |
+
rel="stylesheet",
|
8 |
+
href="https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/5.2.2/css/bootstrap.min.css",
|
9 |
+
),
|
10 |
+
],
|
11 |
+
pico=False,
|
12 |
+
)
|
13 |
+
|
14 |
+
|
15 |
+
@app.get
|
16 |
+
def page():
|
17 |
+
return Div()(
|
18 |
+
Button(
|
19 |
+
"Open Modal",
|
20 |
+
hx_get=modal,
|
21 |
+
hx_target="#modals-here",
|
22 |
+
data_bs_toggle="modal",
|
23 |
+
data_bs_target="#modals-here",
|
24 |
+
cls="btn btn-primary",
|
25 |
+
),
|
26 |
+
Div(id="modals-here", cls="modal modal-blur fade", style="display:none")(
|
27 |
+
Div(cls="modal-dialog modal-lg modal-dialog-centered", role="document")(Div(cls="modal-content"))
|
28 |
+
),
|
29 |
+
)
|
30 |
+
|
31 |
+
|
32 |
+
@app.get
|
33 |
+
def modal():
|
34 |
+
return Div(cls="modal-dialog modal-dialog-centered")(
|
35 |
+
Div(cls="modal-content")(
|
36 |
+
Div(cls="modal-header")(H2("Modal title")),
|
37 |
+
Div(cls="modal-body")("Modal body text goes here."),
|
38 |
+
Div(cls="modal-footer")(Button("Close", cls="btn btn-secondary", data_bs_dismiss="modal")),
|
39 |
+
)
|
40 |
+
)
|
41 |
+
|
42 |
+
|
43 |
+
DESC = "Demonstrates modal dialogs using UIKit"
|
44 |
+
DOC = """
|
45 |
+
Many CSS toolkits include styles (and Javascript) for creating modal dialog boxes. This example shows how to use HTMX alongside original JavaScript provided by Bootstrap.
|
46 |
+
|
47 |
+
We start with a button that triggers the dialog, along with a DIV at the bottom of your markup where the dialog will be loaded:
|
48 |
+
::page::
|
49 |
+
This button uses a GET request to /modal when this button is clicked. The contents of this file will be added to the DOM underneath the #modals-here DIV.
|
50 |
+
|
51 |
+
The server responds with a slightly modified version of Bootstrap’s standard modal
|
52 |
+
::modal::
|
53 |
+
"""
|
54 |
+
HTMX_URL = "https://htmx.org/examples/modal-bootstrap/"
|
55 |
+
HEIGHT = "350px"
|
src/tutorial/htmx/_18_dialogs_custom.py
ADDED
@@ -0,0 +1,50 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import H2, Button, Div, Hr, Link, Script, fast_app
|
2 |
+
|
3 |
+
app, rt = fast_app(
|
4 |
+
hdrs=[
|
5 |
+
Link(rel="stylesheet", href="/htmx_dialogs_custom.css"),
|
6 |
+
Script(src="https://unpkg.com/hyperscript.org", defer=True),
|
7 |
+
],
|
8 |
+
pico=False,
|
9 |
+
)
|
10 |
+
|
11 |
+
|
12 |
+
@app.get
|
13 |
+
def page():
|
14 |
+
return Div()(
|
15 |
+
Button("Open a Modal", hx_get=modal, hx_target="body", hx_swap="beforeend", cls="btn primary"),
|
16 |
+
)
|
17 |
+
|
18 |
+
|
19 |
+
@app.get
|
20 |
+
def modal():
|
21 |
+
return Div(id="modal", _="on closeModal add .closing then wait for animationend then remove me")(
|
22 |
+
Div(cls="modal-underlay", _="on click trigger closeModal"),
|
23 |
+
Div(cls="modal-content")(
|
24 |
+
H2("Modal Dialog"),
|
25 |
+
"This is the modal content. You can put anything here, like text, or a form, or an image.",
|
26 |
+
Hr(),
|
27 |
+
Button("Close", cls="btn danger", _="on click trigger closeModal"),
|
28 |
+
),
|
29 |
+
)
|
30 |
+
|
31 |
+
|
32 |
+
DESC = "Demonstrates modal dialogs from scratch"
|
33 |
+
DOC = """
|
34 |
+
While htmx works great with dialogs built into CSS frameworks (like Bootstrap and UIKit), htmx also makes it easy to build modal dialogs from scratch. Here is a quick example of one way to build them.
|
35 |
+
|
36 |
+
### High Level Plan
|
37 |
+
We’re going to make a button that loads remote content from the server, then displays it in a modal dialog. The modal content will be added to the end of the <body> element, in a div named #modal.
|
38 |
+
|
39 |
+
In this demo we’ll define some nice animations in CSS, and then use some Hyperscript to remove the modals from the DOM when the user is done. Hyperscript is not required with htmx, but the two were designed to be used together and it is much nicer for writing async & event oriented code than JavaScript, which is why we chose it for this example.
|
40 |
+
|
41 |
+
Main Page
|
42 |
+
::page::
|
43 |
+
|
44 |
+
Modal HTML Fragment
|
45 |
+
::modal::
|
46 |
+
|
47 |
+
[Here](/htmx_dialogs_custom.css) is the css for this example.
|
48 |
+
"""
|
49 |
+
HTMX_URL = "https://htmx.org/examples/modal-custom/"
|
50 |
+
HEIGHT = "350px"
|
src/tutorial/htmx/_19_tabs_using_hateoas.py
ADDED
@@ -0,0 +1,53 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Br, Div, Label, Style, fast_app
|
2 |
+
|
3 |
+
css = """
|
4 |
+
.selected {
|
5 |
+
color: rgb(16, 149, 193);
|
6 |
+
cursor: pointer;
|
7 |
+
}
|
8 |
+
"""
|
9 |
+
app, rt = fast_app(hdrs=[Style(css)])
|
10 |
+
|
11 |
+
|
12 |
+
@app.get
|
13 |
+
def page():
|
14 |
+
return Div(hx_get=tab1, hx_trigger="load delay:100ms", hx_target="this")
|
15 |
+
|
16 |
+
|
17 |
+
@app.get
|
18 |
+
def tab1():
|
19 |
+
return make_tab(1)
|
20 |
+
|
21 |
+
|
22 |
+
@app.get
|
23 |
+
def tab2():
|
24 |
+
return make_tab(2)
|
25 |
+
|
26 |
+
|
27 |
+
@app.get
|
28 |
+
def tab3():
|
29 |
+
return make_tab(3)
|
30 |
+
|
31 |
+
|
32 |
+
def make_tab(idx: int):
|
33 |
+
return (
|
34 |
+
Div(style="display:flex;gap:15px")(
|
35 |
+
Label("Tab 1", hx_get=tab1, cls="selected" if idx == 1 else None),
|
36 |
+
Label("Tab 2", hx_get=tab2, cls="selected" if idx == 2 else None),
|
37 |
+
Label("Tab 3", hx_get=tab3, cls="selected" if idx == 3 else None),
|
38 |
+
),
|
39 |
+
Div(f"This is the content of tab {idx}", Br(), str(idx) * 10),
|
40 |
+
)
|
41 |
+
|
42 |
+
|
43 |
+
DESC = "Demonstrates how to display and select tabs using HATEOAS principles"
|
44 |
+
DOC = """
|
45 |
+
This example shows how easy it is to implement tabs using htmx. Following the principle of Hypertext As The Engine Of Application State, the selected tab is a part of the application state. Therefore, to display and select tabs in your application, simply include the tab markup in the returned HTML. If this does not suit your application server design, you can also use a little bit of JavaScript to select tabs instead.
|
46 |
+
|
47 |
+
The main page simply includes the following HTML to load the initial tab into the DOM.
|
48 |
+
::page::
|
49 |
+
Subsequent tab pages display all tabs and highlight the selected one accordingly.
|
50 |
+
::tab1 make_tab::
|
51 |
+
"""
|
52 |
+
HTMX_URL = "https://htmx.org/examples/tabs-hateoas/"
|
53 |
+
HEIGHT = "100px"
|
src/tutorial/htmx/_20_tabs_using_javascript.py
ADDED
@@ -0,0 +1,59 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Div, Label, Style, fast_app
|
2 |
+
|
3 |
+
css = """
|
4 |
+
.selected {
|
5 |
+
color: rgb(16, 149, 193);
|
6 |
+
cursor: pointer;
|
7 |
+
}
|
8 |
+
"""
|
9 |
+
app, rt = fast_app(hdrs=[Style(css)])
|
10 |
+
|
11 |
+
js = """
|
12 |
+
let currentTab = document.querySelector('.selected');
|
13 |
+
currentTab.classList.remove('selected')
|
14 |
+
let newTab = event.target
|
15 |
+
newTab.classList.add('selected')
|
16 |
+
"""
|
17 |
+
|
18 |
+
|
19 |
+
@app.get
|
20 |
+
def page():
|
21 |
+
return (
|
22 |
+
Div(hx_target="#tab_content", **{"hx-on:htmx-after-on-load": js}, style="display:flex;gap:15px")(
|
23 |
+
Label("Tab 1", hx_get=tab1, cls="selected"),
|
24 |
+
Label("Tab 2", hx_get=tab2, cls=""),
|
25 |
+
Label("Tab 3", hx_get=tab3, cls=""),
|
26 |
+
),
|
27 |
+
Div(hx_get=tab1, hx_trigger="load", id="tab_content"),
|
28 |
+
)
|
29 |
+
|
30 |
+
|
31 |
+
@app.get
|
32 |
+
def tab1():
|
33 |
+
return Div("This is the content of tab 1")
|
34 |
+
|
35 |
+
|
36 |
+
@app.get
|
37 |
+
def tab2():
|
38 |
+
return Div("This is the content of tab 2")
|
39 |
+
|
40 |
+
|
41 |
+
@app.get
|
42 |
+
def tab3():
|
43 |
+
return Div("This is the content of tab 3")
|
44 |
+
|
45 |
+
|
46 |
+
DESC = "Demonstrates how to display and select tabs using JavaScript"
|
47 |
+
DOC = """
|
48 |
+
This example shows how to load tab contents using htmx, and to select the “active” tab using Javascript. This reduces some duplication by offloading some of the work of re-rendering the tab HTML from your application server to your clients’ browsers.
|
49 |
+
|
50 |
+
You may also consider a more idiomatic approach that follows the principle of Hypertext As The Engine Of Application State.
|
51 |
+
|
52 |
+
The HTML below displays a list of tabs, with added HTMX to dynamically load each tab pane from the server.
|
53 |
+
::page::
|
54 |
+
|
55 |
+
A simple JavaScript event handler uses the take function to switch the selected tab when the content is swapped into the DOM.
|
56 |
+
::js::
|
57 |
+
"""
|
58 |
+
HTMX_URL = "https://htmx.org/examples/tabs-javascript/"
|
59 |
+
HEIGHT = "100px"
|
src/tutorial/htmx/_21_keyboard_shortcuts.py
ADDED
@@ -0,0 +1,32 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Button, fast_app
|
2 |
+
|
3 |
+
app, rt = fast_app()
|
4 |
+
|
5 |
+
|
6 |
+
# <button class="btn primary" hx-trigger="click, keyup[altKey&&shiftKey&&key=='D'] from:body"
|
7 |
+
# hx-post="/doit">Do It! (alt-shift-D)</button>
|
8 |
+
@app.get
|
9 |
+
def page():
|
10 |
+
return Button("Do It! (shift-D)", hx_trigger="click, keyup[shiftKey&&key=='D'] from:body", hx_post="/doit")
|
11 |
+
|
12 |
+
|
13 |
+
@app.post
|
14 |
+
def doit():
|
15 |
+
return "You did it!"
|
16 |
+
|
17 |
+
|
18 |
+
DESC = "Demonstrates how to create keyboard shortcuts for htmx enabled elements"
|
19 |
+
DOC = """
|
20 |
+
In this example we show how to create a keyboard shortcut for an action.
|
21 |
+
|
22 |
+
We start with a simple button that loads some content from the server:
|
23 |
+
::page::
|
24 |
+
Note that the button responds to both the click event (as usual) and also the keyup event when shift-D is pressed. The from: modifier is used to listen for the keyup event on the body element, thus making it a “global” keyboard shortcut.
|
25 |
+
|
26 |
+
You can trigger the demo below by either clicking on the button, or by hitting shift-D.
|
27 |
+
|
28 |
+
You can find out the conditions needed for a given keyboard shortcut here:
|
29 |
+
|
30 |
+
https://javascript.info/keyboard-events
|
31 |
+
"""
|
32 |
+
HEIGHT = "100px"
|
src/tutorial/htmx/_22_sortable.py
ADDED
@@ -0,0 +1,76 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import time
|
2 |
+
from fasthtml.common import Article, Div, Form, Input, SortableJS, fast_app
|
3 |
+
|
4 |
+
app, rt = fast_app(hdrs=[SortableJS()])
|
5 |
+
|
6 |
+
|
7 |
+
@app.get
|
8 |
+
def page():
|
9 |
+
return (
|
10 |
+
Form(cls="sortable", hx_post="/items", hx_target="#logs", hx_trigger="end")(
|
11 |
+
Div(cls="htmx-indicator")("Updating..."),
|
12 |
+
*[Article(Input(type="hidden", name="item", value=i), f"Item {i}") for i in range(1, 6)],
|
13 |
+
),
|
14 |
+
Div(id="logs"),
|
15 |
+
)
|
16 |
+
|
17 |
+
|
18 |
+
@app.post
|
19 |
+
def items(item: list[str]):
|
20 |
+
time.sleep(0.5)
|
21 |
+
return f"Order: {', '.join(item)}"
|
22 |
+
|
23 |
+
|
24 |
+
DESC = "Demonstrates how to use htmx with the Sortable.js plugin to implement drag-and-drop reordering"
|
25 |
+
DOC = """
|
26 |
+
In this example we show how to integrate the Sortable javascript library with htmx.
|
27 |
+
|
28 |
+
To begin we initialize the `.sortable` class with the Sortable javascript library.
|
29 |
+
This is done automatically by fasthtml SortableJS implementation.
|
30 |
+
|
31 |
+
```python
|
32 |
+
app, rt = fast_app(hdrs=[SortableJS()])
|
33 |
+
```
|
34 |
+
Next, we create a form that has some sortable divs within it, and we trigger an ajax request on the end event, fired by Sortable.js:
|
35 |
+
|
36 |
+
::page::
|
37 |
+
Note that each div has a hidden input inside of it that specifies the item id for that row.
|
38 |
+
When the list is reordered via the Sortable.js drag-and-drop, the end event will be fired. htmx will then post the item ids in the new order to /items, to be persisted by the server.
|
39 |
+
That’s it!
|
40 |
+
|
41 |
+
If you do not use fasthtml SortableJS implementation, you can add the following script to your headers:
|
42 |
+
|
43 |
+
<details>
|
44 |
+
<summary role="button">If you do not use fasthtml SortableJS</summary>
|
45 |
+
|
46 |
+
```javascript
|
47 |
+
htmx.onLoad(function(content) {
|
48 |
+
var sortables = content.querySelectorAll(".sortable");
|
49 |
+
for (var i = 0; i < sortables.length; i++) {
|
50 |
+
var sortable = sortables[i];
|
51 |
+
var sortableInstance = new Sortable(sortable, {
|
52 |
+
animation: 150,
|
53 |
+
ghostClass: 'blue-background-class',
|
54 |
+
|
55 |
+
// Make the `.htmx-indicator` unsortable
|
56 |
+
filter: ".htmx-indicator",
|
57 |
+
onMove: function (evt) {
|
58 |
+
return evt.related.className.indexOf('htmx-indicator') === -1;
|
59 |
+
},
|
60 |
+
|
61 |
+
// Disable sorting on the `end` event
|
62 |
+
onEnd: function (evt) {
|
63 |
+
this.option("disabled", true);
|
64 |
+
}
|
65 |
+
});
|
66 |
+
|
67 |
+
// Re-enable sorting on the `htmx:afterSwap` event
|
68 |
+
sortable.addEventListener("htmx:afterSwap", function() {
|
69 |
+
sortableInstance.option("disabled", false);
|
70 |
+
});
|
71 |
+
}
|
72 |
+
})
|
73 |
+
```
|
74 |
+
</details>
|
75 |
+
"""
|
76 |
+
# HEIGHT = "100px"
|
src/tutorial/htmx/_23_update_other_content.py
ADDED
@@ -0,0 +1,239 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# fmt: off
|
2 |
+
from fasthtml.common import H3, Div, Form, Input, Label, Script, Style, Table, Tbody, Td, Th, Thead, Tr, fast_app
|
3 |
+
# fmt: on
|
4 |
+
|
5 |
+
css = """
|
6 |
+
.selected {
|
7 |
+
color: rgb(16, 149, 193);
|
8 |
+
cursor: pointer;
|
9 |
+
}
|
10 |
+
"""
|
11 |
+
app, rt = fast_app(hdrs=[Style(css)])
|
12 |
+
|
13 |
+
data = [
|
14 |
+
("phi", "[email protected]"),
|
15 |
+
("grace", "[email protected]"),
|
16 |
+
]
|
17 |
+
|
18 |
+
|
19 |
+
@app.get
|
20 |
+
def page():
|
21 |
+
return Div(cls="container")(
|
22 |
+
Div(hx_get=solution.rt(idx=1), hx_trigger="load delay:100ms", hx_target="this", hx_swap="innerHTML")
|
23 |
+
)
|
24 |
+
|
25 |
+
|
26 |
+
@app.get
|
27 |
+
def solution(idx: int):
|
28 |
+
return Div(
|
29 |
+
Div(
|
30 |
+
Div(style="display:flex;gap:15px")(
|
31 |
+
Label("Solution 1", hx_get=solution.rt(idx=1), cls="selected" if idx == 1 else None),
|
32 |
+
Label("Solution 2", hx_get=solution.rt(idx=2), cls="selected" if idx == 2 else None),
|
33 |
+
Label("Solution 3", hx_get=solution.rt(idx=3), cls="selected" if idx == 3 else None),
|
34 |
+
Label("Solution 4", hx_get=solution.rt(idx=4), cls="selected" if idx == 4 else None),
|
35 |
+
),
|
36 |
+
),
|
37 |
+
Div(
|
38 |
+
H3(f"Solution {idx}"),
|
39 |
+
[solution1, solution2, solution3, solution4][idx - 1](),
|
40 |
+
),
|
41 |
+
)
|
42 |
+
|
43 |
+
|
44 |
+
# ---- Solution 1 ----
|
45 |
+
|
46 |
+
|
47 |
+
def solution1():
|
48 |
+
return Div(hx_target="this")(
|
49 |
+
Table(
|
50 |
+
Thead(Tr(Th("Name"), Th("Email"), Th())),
|
51 |
+
Tbody(*[Tr(Td(d) for d in row) for row in data]),
|
52 |
+
),
|
53 |
+
Form(form_fields(), hx_post=solution1_add_contact, hx_swap="outerHTML"),
|
54 |
+
)
|
55 |
+
|
56 |
+
|
57 |
+
@app.post("/solution1/contacts")
|
58 |
+
def solution1_add_contact(name: str, email: str):
|
59 |
+
add_contact(name, email)
|
60 |
+
return solution1()
|
61 |
+
|
62 |
+
|
63 |
+
# ---- Solution 2 ----
|
64 |
+
|
65 |
+
|
66 |
+
def solution2():
|
67 |
+
return Div()(
|
68 |
+
Table(
|
69 |
+
Thead(Tr(Th("Name"), Th("Email"), Th())),
|
70 |
+
Tbody(id="table")(*[Tr(Td(d) for d in row) for row in data]),
|
71 |
+
),
|
72 |
+
Form(form_fields(), hx_post=solution2_add_contact, hx_target="this"),
|
73 |
+
)
|
74 |
+
|
75 |
+
|
76 |
+
@app.post("/solution2/contacts")
|
77 |
+
def solution2_add_contact(name: str, email: str):
|
78 |
+
add_contact(name, email)
|
79 |
+
return (
|
80 |
+
Tbody(hx_swap_oob="beforeend:#table")(Tr(Td(name), Td(email))),
|
81 |
+
form_fields(),
|
82 |
+
)
|
83 |
+
|
84 |
+
|
85 |
+
# ---- Solution 3 ----
|
86 |
+
|
87 |
+
|
88 |
+
def solution3():
|
89 |
+
return Div()(
|
90 |
+
Div(
|
91 |
+
Table(
|
92 |
+
Thead(Tr(Th("Name"), Th("Email"), Th())),
|
93 |
+
Tbody(id="table", hx_get=solution3_contacts_table, hx_trigger="newContact from:body", hx_target="this")(
|
94 |
+
*[Tr(Td(d) for d in row) for row in data]
|
95 |
+
),
|
96 |
+
),
|
97 |
+
),
|
98 |
+
Form(form_fields(), hx_post=solution3_add_contact, hx_target="this"),
|
99 |
+
)
|
100 |
+
|
101 |
+
|
102 |
+
@app.get("/solution3/contacts/table")
|
103 |
+
def solution3_contacts_table():
|
104 |
+
return [Tr(Td(d) for d in row) for row in data]
|
105 |
+
|
106 |
+
|
107 |
+
@app.post("/solution3/contacts")
|
108 |
+
def solution3_add_contact(name: str, email: str):
|
109 |
+
add_contact(name, email)
|
110 |
+
return (
|
111 |
+
form_fields(),
|
112 |
+
Script("htmx.trigger('body', 'newContact')"),
|
113 |
+
)
|
114 |
+
|
115 |
+
|
116 |
+
# ---- Solution 4 ----
|
117 |
+
|
118 |
+
|
119 |
+
def solution4():
|
120 |
+
return (
|
121 |
+
Script(src="https://unpkg.com/[email protected]/path-deps.js"),
|
122 |
+
Div(hx_ext="path-deps")(
|
123 |
+
Table(
|
124 |
+
Thead(Tr(Th("Name"), Th("Email"), Th())),
|
125 |
+
Tbody(
|
126 |
+
hx_get=solution4_contacts_table,
|
127 |
+
hx_trigger="path-deps",
|
128 |
+
path_deps="/update-other-content/solution4/contacts",
|
129 |
+
hx_target="this",
|
130 |
+
)(*[Tr(Td(d) for d in row) for row in data]),
|
131 |
+
),
|
132 |
+
Form(form_fields(), hx_post=solution4_add_contact, hx_target="this"),
|
133 |
+
),
|
134 |
+
)
|
135 |
+
|
136 |
+
|
137 |
+
@app.get("/solution4/contacts/table")
|
138 |
+
def solution4_contacts_table():
|
139 |
+
return [Tr(Td(d) for d in row) for row in data]
|
140 |
+
|
141 |
+
|
142 |
+
@app.post("/solution4/contacts")
|
143 |
+
def solution4_add_contact(name: str, email: str):
|
144 |
+
add_contact(name, email)
|
145 |
+
return form_fields()
|
146 |
+
|
147 |
+
|
148 |
+
# ---- Utilities ----
|
149 |
+
|
150 |
+
|
151 |
+
def form_fields():
|
152 |
+
return Div(style="display:flex;gap:5px")(
|
153 |
+
Label("Name", Input(name="name")), Label("Email", Input(name="email")), Input(type="submit", hidden=True)
|
154 |
+
)
|
155 |
+
|
156 |
+
|
157 |
+
def add_contact(name, email):
|
158 |
+
global data
|
159 |
+
data.append([name, email])
|
160 |
+
data = data[-5:]
|
161 |
+
|
162 |
+
|
163 |
+
DESC = "Demonstrates how to update content beyond just the target elements"
|
164 |
+
DOC = """
|
165 |
+
A question that often comes up when people are first working with htmx is:
|
166 |
+
> I need to update other content on the screen. How do I do this?
|
167 |
+
|
168 |
+
There are multiple ways to do so, and in this example we will walk you through some of them.
|
169 |
+
|
170 |
+
We’ll use the following basic UI to discuss this concept: a simple table of contacts, and a form below it to add new contacts to the table using hx-post.
|
171 |
+
|
172 |
+
The problem here is that when you submit a new contact in the form, you want the contact table above to refresh and include the contact that was just added by the form.
|
173 |
+
|
174 |
+
What solutions do we have?
|
175 |
+
|
176 |
+
## Solution 1: Expand the Target
|
177 |
+
|
178 |
+
The easiest solution here is to ���expand the target” of the form to enclose both the table and the form. For example, you could wrap the whole thing in a div and then target that div in the form:
|
179 |
+
::solution1 solution1_add_contact::
|
180 |
+
|
181 |
+
Note that we are targeting the enclosing div using the hx-target attribute. You would need to render both the table and the form in the response to the POST to /contacts.
|
182 |
+
|
183 |
+
This is a simple and reliable approach, although it might not feel particularly elegant.
|
184 |
+
|
185 |
+
## Solution 2: Out of Band Responses
|
186 |
+
|
187 |
+
A more sophisticated approach to this problem would use out of band swaps to swap in updated content to the DOM.
|
188 |
+
|
189 |
+
Using this approach, the HTML doesn’t need to change from the original setup at all:
|
190 |
+
|
191 |
+
::solution2::
|
192 |
+
|
193 |
+
Instead of modifying something on the front end, in your response to the POST to /contacts you would include some additional content:
|
194 |
+
|
195 |
+
::solution2_add_contact::
|
196 |
+
|
197 |
+
This content uses the [hx-swap-oob](https://htmx.org/attributes/hx-swap-oob/) attribute to append itself to the #table, updating the table after a contact is added successfully.
|
198 |
+
|
199 |
+
## Solution 3: Triggering Events
|
200 |
+
|
201 |
+
An even more sophisticated approach would be to trigger a client side event when a successful contact is created and then listen for that event on the table, causing the table to refresh.
|
202 |
+
|
203 |
+
::solution3::
|
204 |
+
|
205 |
+
We have added a new end-point `/contacts/table` that re-renders the contacts table. Our trigger for this request is a custom event, newContact. We listen for this event on the body because when it is triggered by the response to the form, it will end up hitting the body due to event bubbling.
|
206 |
+
|
207 |
+
::solution3_contacts_table::
|
208 |
+
|
209 |
+
When a successful contact creation occurs during a POST to `/contacts`, the response includes an `HX-Trigger` response header that looks like this:
|
210 |
+
|
211 |
+
::solution3_add_contact::
|
212 |
+
|
213 |
+
`HX-Trigger:newContact`
|
214 |
+
This will trigger the table to issue a GET to `/contacts/table` and this will render the newly added contact row
|
215 |
+
(in addition to the rest of the table.)
|
216 |
+
|
217 |
+
Very clean, event driven programming!
|
218 |
+
|
219 |
+
## Solution 4: Using the Path Dependencies Extension
|
220 |
+
|
221 |
+
A final approach is to use REST-ful path dependencies to refresh the table. Intercooler.js, the predecessor to htmx, had path-based dependencies integrated into the library.
|
222 |
+
|
223 |
+
htmx dropped this as a core feature, but supports an extension, [path deps](https://github.com/bigskysoftware/htmx-extensions/blob/main/src/path-deps/README.md), that gives you similar functionality.
|
224 |
+
|
225 |
+
Updating our example to use the extension would involve loading the extension javascript and then annotating our HTML like so:
|
226 |
+
|
227 |
+
::solution4 solution4_add_contact::
|
228 |
+
Now, when the form posts to the /contacts URL, the path-deps extension will detect that and trigger an path-deps event on the contacts table, therefore triggering a request.
|
229 |
+
|
230 |
+
The advantage here is that you don’t need to do anything fancy with response headers. The downside is that a request will be issued on every POST, even if a contact was not successfully created.
|
231 |
+
|
232 |
+
## Which should I use?
|
233 |
+
Generally I would recommend the first approach, expanding your target, especially if the elements that need to be updated are reasonably close to one another in the DOM. It is simple and reliable.
|
234 |
+
|
235 |
+
After that, I would say it is a tossup between the custom event and an OOB swap approaches. I would lean towards the custom event approach because I like event-oriented systems, but that’s a personal preference. Which one you choose should be dictated by your own software engineering tastes and which of the two matches up better with your server side technology of choice.
|
236 |
+
|
237 |
+
Finally, the path-deps approach is interesting, and if it fits well with your mental model and overall system architecture, it can be a fun way to avoid explicit refreshing. I would look at it last, however, unless the concept really grabs you.
|
238 |
+
"""
|
239 |
+
HEIGHT = "400px"
|
src/tutorial/htmx/_24_confirm_custom.py
ADDED
@@ -0,0 +1,82 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Div, fast_app, Script, Button, Label
|
2 |
+
|
3 |
+
js_solution_2 = """
|
4 |
+
document.addEventListener("htmx:confirm", function(e) {
|
5 |
+
console.log(e);
|
6 |
+
// The event is triggered on every trigger for a request, so we need to check if the element
|
7 |
+
// that triggered the request has a hx-confirm attribute, if not we can return early and let
|
8 |
+
// the default behavior happen
|
9 |
+
if (!e.detail.target.hasAttribute('hx-confirm')) return
|
10 |
+
|
11 |
+
// This will prevent the request from being issued to later manually issue it
|
12 |
+
e.preventDefault()
|
13 |
+
|
14 |
+
Swal.fire({
|
15 |
+
title: "Proceed?",
|
16 |
+
text: `I ask you... ${e.detail.question}`
|
17 |
+
}).then(function(result) {
|
18 |
+
if (result.isConfirmed) {
|
19 |
+
// If the user confirms, we manually issue the request
|
20 |
+
e.detail.issueRequest(true); // true to skip the built-in window.confirm()
|
21 |
+
}
|
22 |
+
})
|
23 |
+
})
|
24 |
+
"""
|
25 |
+
|
26 |
+
app, rt = fast_app(hdrs=[Script(src="https://cdn.jsdelivr.net/npm/sweetalert2@11"), Script(js_solution_2)])
|
27 |
+
|
28 |
+
|
29 |
+
@app.get
|
30 |
+
def page():
|
31 |
+
return Div(cls="container grid")(Label("Solution 1"), solution1(), Label("Solution 2"), solution2())
|
32 |
+
|
33 |
+
|
34 |
+
# ---- Solution 1 ----
|
35 |
+
def solution1():
|
36 |
+
js = """
|
37 |
+
Swal.fire({title: 'Confirm', text:'Do you want to continue?'}).then((result)=>{
|
38 |
+
if(result.isConfirmed){
|
39 |
+
htmx.trigger(this, 'confirmed');
|
40 |
+
}
|
41 |
+
})
|
42 |
+
"""
|
43 |
+
return Button("Click me", hx_get=confirmed, hx_trigger="confirmed", onClick=js)
|
44 |
+
|
45 |
+
|
46 |
+
@app.get
|
47 |
+
def confirmed():
|
48 |
+
return "Confirmed"
|
49 |
+
|
50 |
+
|
51 |
+
# ---- Solution 2 ----
|
52 |
+
|
53 |
+
|
54 |
+
def solution2():
|
55 |
+
return Button("Click me", hx_get=confirmed, hx_confirm="Do you want to continue?")
|
56 |
+
|
57 |
+
|
58 |
+
DESC = "Demonstrates how to implement a custom confirmation dialog with htmx"
|
59 |
+
DOC = """
|
60 |
+
htmx supports the [hx-confirm](https://htmx.org/attributes/hx-confirm/) attribute to provide a simple mechanism for confirming a user action. This uses the default confirm() function in javascript which, while trusty, may not be consistent with your applications UX.
|
61 |
+
|
62 |
+
In this example we will see how to use [sweetalert2](https://sweetalert2.github.io/) to implement a custom confirmation dialog. Below are two examples, one using a click+custom event method, and one using the built-in hx-confirm attribute and the [htmx:confirm](https://htmx.org/events/#htmx:confirm) event.
|
63 |
+
|
64 |
+
## Using on click+custom event
|
65 |
+
::solution1::
|
66 |
+
|
67 |
+
Here we use javascript to show a Sweet Alert 2 on a click, asking for confirmation. If the user confirms the dialog, we then trigger the request by triggering the custom “confirmed” event which is then picked up by hx-trigger.
|
68 |
+
|
69 |
+
## Vanilla JS, hx-confirm
|
70 |
+
|
71 |
+
We add some javascript to invoke Sweet Alert 2 on a click, asking for confirmation. If the user confirms the dialog, we trigger the request by calling the issueRequest method. We pass `skipConfirmation=true` as argument to skip `window.confirm`.
|
72 |
+
|
73 |
+
::solution2::
|
74 |
+
Javascript code
|
75 |
+
|
76 |
+
::js_solution_2::
|
77 |
+
This allows to use hx-confirm’s value in the prompt which is convenient when the question depends on the element.
|
78 |
+
|
79 |
+
Learn more about the htmx:confirm event [here](https://htmx.org/events/#htmx:confirm).
|
80 |
+
"""
|
81 |
+
HTMX_URL = "https://htmx.org/examples/confirm/"
|
82 |
+
HEIGHT = "250px"
|
src/tutorial/htmx/_25_async_auth.py
ADDED
@@ -0,0 +1,65 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import Div, fast_app, Button, Output, Script
|
2 |
+
|
3 |
+
js = """
|
4 |
+
// auth is a promise returned by our authentication system
|
5 |
+
const auth = new Promise((resolve, reject) => {
|
6 |
+
setTimeout(() => {
|
7 |
+
resolve("super-dummy-token");
|
8 |
+
}, 300);
|
9 |
+
});
|
10 |
+
|
11 |
+
// await the auth token and store it somewhere
|
12 |
+
let authToken = null;
|
13 |
+
auth.then((token) => {
|
14 |
+
authToken = token
|
15 |
+
})
|
16 |
+
|
17 |
+
// gate htmx requests on the auth token
|
18 |
+
htmx.on("htmx:confirm", (e)=> {
|
19 |
+
// if there is no auth token
|
20 |
+
if(authToken == null) {
|
21 |
+
// stop the regular request from being issued
|
22 |
+
e.preventDefault()
|
23 |
+
// only issue it once the auth promise has resolved
|
24 |
+
auth.then(() => e.detail.issueRequest())
|
25 |
+
}
|
26 |
+
})
|
27 |
+
|
28 |
+
// add the auth token to the request as a header
|
29 |
+
htmx.on("htmx:configRequest", (e)=> {
|
30 |
+
e.detail.headers["AUTH"] = authToken
|
31 |
+
})
|
32 |
+
"""
|
33 |
+
|
34 |
+
app, rt = fast_app(hdrs=[Script(js)])
|
35 |
+
|
36 |
+
|
37 |
+
@app.get
|
38 |
+
def page():
|
39 |
+
return Div(cls="container grid")(
|
40 |
+
Button("An htmx-Powered button", hx_post=example, hx_target="next output"), Output("--")
|
41 |
+
)
|
42 |
+
|
43 |
+
|
44 |
+
@app.post
|
45 |
+
def example(request):
|
46 |
+
return "Method call with token: " + request.headers["AUTH"]
|
47 |
+
|
48 |
+
|
49 |
+
DESC = "Demonstrates how to handle async authentication tokens in htmx"
|
50 |
+
DOC = """
|
51 |
+
This example shows how to implement an an async auth token flow for htmx.
|
52 |
+
|
53 |
+
The technique we will use here will take advantage of the fact that you can delay requests using the htmx:confirm event.
|
54 |
+
|
55 |
+
We first have a button that should not issue a request until an auth token has been retrieved:
|
56 |
+
|
57 |
+
::page example::
|
58 |
+
Next we will add some scripting to work with an auth promise (returned by a library):
|
59 |
+
|
60 |
+
::js::
|
61 |
+
Here we use a global variable, but you could use localStorage or whatever preferred mechanism you want to communicate the authentication token to the `htmx:configRequest` event.
|
62 |
+
|
63 |
+
With this code in place, htmx will not issue requests until the auth promise has been resolved.
|
64 |
+
"""
|
65 |
+
HEIGHT = "150px"
|
src/tutorial/htmx/_26_web_components.py
ADDED
@@ -0,0 +1,50 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import fast_app, Script, Article, ft_hx
|
2 |
+
|
3 |
+
js = """
|
4 |
+
customElements.define('my-component', class MyComponent extends HTMLElement {
|
5 |
+
// This method runs when your custom element is added to the page
|
6 |
+
connectedCallback() {
|
7 |
+
const root = this.attachShadow({ mode: 'closed' })
|
8 |
+
root.innerHTML = `
|
9 |
+
<button hx-get="/my-component-clicked" hx-target="next div">Click me!</button>
|
10 |
+
<div></div>
|
11 |
+
`
|
12 |
+
htmx.process(root) // Tell HTMX about this component's shadow DOM
|
13 |
+
}
|
14 |
+
})
|
15 |
+
"""
|
16 |
+
|
17 |
+
app, rt = fast_app(hdrs=[Script(js)])
|
18 |
+
|
19 |
+
|
20 |
+
@app.get
|
21 |
+
def page():
|
22 |
+
MyComponent = ft_hx("my-component")
|
23 |
+
# Alternative: NotStr(MyComponent())
|
24 |
+
return Article(MyComponent())
|
25 |
+
|
26 |
+
|
27 |
+
@app.get("/my-component-clicked")
|
28 |
+
def example():
|
29 |
+
return "Clicked!"
|
30 |
+
|
31 |
+
|
32 |
+
DESC = "Demonstrates how to integrate htmx with web components and shadow DOM"
|
33 |
+
DOC = """
|
34 |
+
This example shows how to integrate HTMX with web components, allowing it to be used inside of shadow DOM.
|
35 |
+
|
36 |
+
By default, HTMX doesn’t know anything about your web components, and won’t see anything inside their shadow DOM. Because of this, you’ll need to manually tell HTMX about your component’s shadow DOM by using [htmx.process](https://htmx.org/api/#process).
|
37 |
+
|
38 |
+
Define a custom component (javascript)
|
39 |
+
::js::
|
40 |
+
|
41 |
+
Usage (python)
|
42 |
+
::page::
|
43 |
+
|
44 |
+
Once you’ve told HTMX about your component’s shadow DOM, most things should work as expected. However, note that selectors such as in hx-target will only see elements inside the same shadow DOM - if you need to access things outside of your web components, you can use one of the following options:
|
45 |
+
|
46 |
+
host: Selects the element hosting the current shadow DOM
|
47 |
+
global: If used as a prefix, selects from the main document instead of the current shadow DOM
|
48 |
+
The same principles generally apply to web components that don’t use shadow DOM as well; while selectors won’t be encapsulated like with shadow DOM, you’ll still have to point HTMX to your component’s content by calling htmx.process.
|
49 |
+
"""
|
50 |
+
HEIGHT = "100px"
|
src/tutorial/htmx/_27_move_before.py
ADDED
@@ -0,0 +1,60 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fasthtml.common import fast_app, Div, Iframe, A, Figure, Label, Article
|
2 |
+
|
3 |
+
|
4 |
+
app, rt = fast_app()
|
5 |
+
|
6 |
+
|
7 |
+
@app.get
|
8 |
+
def page():
|
9 |
+
return Div(
|
10 |
+
Label("Feel free to change the page back and forth. The video will keep playing. Enjoy!"),
|
11 |
+
Iframe(hx_preserve=True, id="rick-roll", **iframe_kv),
|
12 |
+
Label("You are on page: /page1"),
|
13 |
+
A("Go to page 2", hx_boost=True, hx_get=page2, hx_target="body"),
|
14 |
+
)
|
15 |
+
|
16 |
+
|
17 |
+
@app.get
|
18 |
+
def page2():
|
19 |
+
return Article(
|
20 |
+
Label("Feel free to change the page back and forth. The video will keep playing. Enjoy!"),
|
21 |
+
Figure(
|
22 |
+
Iframe(hx_preserve=True, id="rick-roll", **iframe_kv),
|
23 |
+
),
|
24 |
+
Label("You are on page: /page2"),
|
25 |
+
A("Back to page 1", hx_boost=True, hx_get=page, hx_target="body"),
|
26 |
+
)
|
27 |
+
|
28 |
+
|
29 |
+
iframe_kv = dict(
|
30 |
+
src="https://www.youtube.com/embed/GFq6wH5JR2A",
|
31 |
+
title="Rick Astley - Never Gonna Give You Up (Official Music Video)",
|
32 |
+
frameborder="0",
|
33 |
+
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share",
|
34 |
+
referrerpolicy="strict-origin-when-cross-origin",
|
35 |
+
allowfullscreen="",
|
36 |
+
)
|
37 |
+
|
38 |
+
|
39 |
+
DESC = "htmx will use the experimental moveBefore() API for moving elements, if it is present."
|
40 |
+
DOC = """
|
41 |
+
This page demonstrates the use of the experimental [moveBefore()](https://github.com/whatwg/dom/issues/1255) DOM API available in Chrome Canary, which has been integrated into the hx-preserve attribute of htmx, if it is available.
|
42 |
+
|
43 |
+
## Getting Chrome Canary & Enabling moveBefore()
|
44 |
+
For the demo to work properly you will need to install Chrome Canary and enable the API:
|
45 |
+
|
46 |
+
Navigate to [chrome://flags](chrome://flags).
|
47 |
+
Enable “Atomic DOM move”.
|
48 |
+
|
49 |
+
htmx takes advantage of this API in the `hx-preserve` functionality if it is available, allowing you to mark an element as “preserved” and having all its state preserved as it moves between areas on the screen when new content is merged in. This is significantly better developer experience than current alternatives, such as morphing, which rely on the structure of the new page being “close enough” to not have to move things like video elements.
|
50 |
+
|
51 |
+
## Demo
|
52 |
+
The video continues to play after changing of pages
|
53 |
+
|
54 |
+
::page page2::
|
55 |
+
|
56 |
+
If you inspect the video below you will see that it is embedded in a div element. If you click the “Go to page 2” link, which is boosted, you will transition to another page with a video element with the same id, but embedded in a figure element instead. Without the moveBefore() functionality it is impossible to keep the video playing in this situation because “reparenting” (i.e. changing the parent of an element) causes it to reset.
|
57 |
+
|
58 |
+
moveBefore() opens up a huge number of possibilities in web development by allowing developers to completely change the layout of a page while still preserving elements play state, focus, etc.
|
59 |
+
"""
|
60 |
+
HEIGHT = "300px"
|
uv.lock
CHANGED
@@ -2,17 +2,7 @@ version = 1
|
|
2 |
requires-python = ">=3.11"
|
3 |
resolution-markers = [
|
4 |
"python_full_version < '3.12'",
|
5 |
-
"python_full_version
|
6 |
-
"python_full_version >= '3.13'",
|
7 |
-
]
|
8 |
-
|
9 |
-
[[package]]
|
10 |
-
name = "annotated-types"
|
11 |
-
version = "0.7.0"
|
12 |
-
source = { registry = "https://pypi.org/simple" }
|
13 |
-
sdist = { url = "https://files.pythonhosted.org/packages/ee/67/531ea369ba64dcff5ec9c3402f9f51bf748cec26dde048a2f973a4eea7f5/annotated_types-0.7.0.tar.gz", hash = "sha256:aff07c09a53a08bc8cfccb9c85b05f1aa9a2a6f23728d790723543408344ce89", size = 16081 }
|
14 |
-
wheels = [
|
15 |
-
{ url = "https://files.pythonhosted.org/packages/78/b6/6307fbef88d9b5ee7421e68d78a9f162e0da4900bc5f5793f6d3d0e34fb8/annotated_types-0.7.0-py3-none-any.whl", hash = "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53", size = 13643 },
|
16 |
]
|
17 |
|
18 |
[[package]]
|
@@ -117,41 +107,56 @@ wheels = [
|
|
117 |
|
118 |
[[package]]
|
119 |
name = "charset-normalizer"
|
120 |
-
version = "3.
|
121 |
-
source = { registry = "https://pypi.org/simple" }
|
122 |
-
sdist = { url = "https://files.pythonhosted.org/packages/
|
123 |
-
wheels = [
|
124 |
-
{ url = "https://files.pythonhosted.org/packages/
|
125 |
-
{ url = "https://files.pythonhosted.org/packages/
|
126 |
-
{ url = "https://files.pythonhosted.org/packages/
|
127 |
-
{ url = "https://files.pythonhosted.org/packages/
|
128 |
-
{ url = "https://files.pythonhosted.org/packages/
|
129 |
-
{ url = "https://files.pythonhosted.org/packages/
|
130 |
-
{ url = "https://files.pythonhosted.org/packages/
|
131 |
-
{ url = "https://files.pythonhosted.org/packages/
|
132 |
-
{ url = "https://files.pythonhosted.org/packages/
|
133 |
-
{ url = "https://files.pythonhosted.org/packages/
|
134 |
-
{ url = "https://files.pythonhosted.org/packages/
|
135 |
-
{ url = "https://files.pythonhosted.org/packages/
|
136 |
-
{ url = "https://files.pythonhosted.org/packages/
|
137 |
-
{ url = "https://files.pythonhosted.org/packages/
|
138 |
-
{ url = "https://files.pythonhosted.org/packages/
|
139 |
-
{ url = "https://files.pythonhosted.org/packages/
|
140 |
-
{ url = "https://files.pythonhosted.org/packages/
|
141 |
-
{ url = "https://files.pythonhosted.org/packages/
|
142 |
-
{ url = "https://files.pythonhosted.org/packages/
|
143 |
-
{ url = "https://files.pythonhosted.org/packages/
|
144 |
-
{ url = "https://files.pythonhosted.org/packages/
|
145 |
-
{ url = "https://files.pythonhosted.org/packages/
|
146 |
-
{ url = "https://files.pythonhosted.org/packages/
|
147 |
-
{ url = "https://files.pythonhosted.org/packages/
|
148 |
-
{ url = "https://files.pythonhosted.org/packages/
|
149 |
-
{ url = "https://files.pythonhosted.org/packages/
|
150 |
-
{ url = "https://files.pythonhosted.org/packages/
|
151 |
-
{ url = "https://files.pythonhosted.org/packages/
|
152 |
-
{ url = "https://files.pythonhosted.org/packages/
|
153 |
-
{ url = "https://files.pythonhosted.org/packages/
|
154 |
-
{ url = "https://files.pythonhosted.org/packages/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
155 |
]
|
156 |
|
157 |
[[package]]
|
@@ -189,50 +194,50 @@ wheels = [
|
|
189 |
|
190 |
[[package]]
|
191 |
name = "coverage"
|
192 |
-
version = "7.6.
|
193 |
-
source = { registry = "https://pypi.org/simple" }
|
194 |
-
sdist = { url = "https://files.pythonhosted.org/packages/
|
195 |
-
wheels = [
|
196 |
-
{ url = "https://files.pythonhosted.org/packages/
|
197 |
-
{ url = "https://files.pythonhosted.org/packages/
|
198 |
-
{ url = "https://files.pythonhosted.org/packages/
|
199 |
-
{ url = "https://files.pythonhosted.org/packages/
|
200 |
-
{ url = "https://files.pythonhosted.org/packages/
|
201 |
-
{ url = "https://files.pythonhosted.org/packages/
|
202 |
-
{ url = "https://files.pythonhosted.org/packages/
|
203 |
-
{ url = "https://files.pythonhosted.org/packages/
|
204 |
-
{ url = "https://files.pythonhosted.org/packages/
|
205 |
-
{ url = "https://files.pythonhosted.org/packages/
|
206 |
-
{ url = "https://files.pythonhosted.org/packages/
|
207 |
-
{ url = "https://files.pythonhosted.org/packages/
|
208 |
-
{ url = "https://files.pythonhosted.org/packages/
|
209 |
-
{ url = "https://files.pythonhosted.org/packages/
|
210 |
-
{ url = "https://files.pythonhosted.org/packages/
|
211 |
-
{ url = "https://files.pythonhosted.org/packages/
|
212 |
-
{ url = "https://files.pythonhosted.org/packages/
|
213 |
-
{ url = "https://files.pythonhosted.org/packages/
|
214 |
-
{ url = "https://files.pythonhosted.org/packages/
|
215 |
-
{ url = "https://files.pythonhosted.org/packages/
|
216 |
-
{ url = "https://files.pythonhosted.org/packages/
|
217 |
-
{ url = "https://files.pythonhosted.org/packages/
|
218 |
-
{ url = "https://files.pythonhosted.org/packages/
|
219 |
-
{ url = "https://files.pythonhosted.org/packages/
|
220 |
-
{ url = "https://files.pythonhosted.org/packages/
|
221 |
-
{ url = "https://files.pythonhosted.org/packages/
|
222 |
-
{ url = "https://files.pythonhosted.org/packages/
|
223 |
-
{ url = "https://files.pythonhosted.org/packages/
|
224 |
-
{ url = "https://files.pythonhosted.org/packages/
|
225 |
-
{ url = "https://files.pythonhosted.org/packages/
|
226 |
-
{ url = "https://files.pythonhosted.org/packages/
|
227 |
-
{ url = "https://files.pythonhosted.org/packages/
|
228 |
-
{ url = "https://files.pythonhosted.org/packages/
|
229 |
-
{ url = "https://files.pythonhosted.org/packages/
|
230 |
-
{ url = "https://files.pythonhosted.org/packages/
|
231 |
-
{ url = "https://files.pythonhosted.org/packages/
|
232 |
-
{ url = "https://files.pythonhosted.org/packages/
|
233 |
-
{ url = "https://files.pythonhosted.org/packages/
|
234 |
-
{ url = "https://files.pythonhosted.org/packages/
|
235 |
-
{ url = "https://files.pythonhosted.org/packages/
|
236 |
]
|
237 |
|
238 |
[package.optional-dependencies]
|
@@ -266,41 +271,6 @@ wheels = [
|
|
266 |
{ url = "https://files.pythonhosted.org/packages/d5/50/83c593b07763e1161326b3b8c6686f0f4b0f24d5526546bee538c89837d6/decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186", size = 9073 },
|
267 |
]
|
268 |
|
269 |
-
[[package]]
|
270 |
-
name = "devicorn"
|
271 |
-
version = "0.1.0"
|
272 |
-
source = { editable = "../devicorn" }
|
273 |
-
dependencies = [
|
274 |
-
{ name = "pydantic" },
|
275 |
-
{ name = "pyinstrument" },
|
276 |
-
{ name = "tomli" },
|
277 |
-
{ name = "typer" },
|
278 |
-
{ name = "uvicorn" },
|
279 |
-
{ name = "watchfiles" },
|
280 |
-
]
|
281 |
-
|
282 |
-
[package.metadata]
|
283 |
-
requires-dist = [
|
284 |
-
{ name = "pydantic", specifier = ">=2.9.2" },
|
285 |
-
{ name = "pyinstrument", specifier = ">=4.7.3" },
|
286 |
-
{ name = "tomli", specifier = ">=1" },
|
287 |
-
{ name = "typer", specifier = ">=0.12.5" },
|
288 |
-
{ name = "uvicorn", specifier = ">=0.31.0" },
|
289 |
-
{ name = "watchfiles", specifier = ">=0.24.0" },
|
290 |
-
]
|
291 |
-
|
292 |
-
[package.metadata.requires-dev]
|
293 |
-
dev = [
|
294 |
-
{ name = "fastapi", specifier = ">=0.115.0" },
|
295 |
-
{ name = "inline-snapshot", specifier = ">=0.13.3" },
|
296 |
-
{ name = "ipykernel", specifier = ">=6.29.5" },
|
297 |
-
{ name = "pytest", specifier = ">=8.3.3" },
|
298 |
-
{ name = "pytest-asyncio", specifier = ">=0.24.0" },
|
299 |
-
{ name = "pytest-cov", specifier = ">=5.0.0" },
|
300 |
-
{ name = "python-fasthtml", specifier = ">=0.6.9" },
|
301 |
-
{ name = "ruff", specifier = ">=0.6.8" },
|
302 |
-
]
|
303 |
-
|
304 |
[[package]]
|
305 |
name = "diskcache"
|
306 |
version = "5.6.3"
|
@@ -321,14 +291,14 @@ wheels = [
|
|
321 |
|
322 |
[[package]]
|
323 |
name = "fastcore"
|
324 |
-
version = "1.7.
|
325 |
source = { registry = "https://pypi.org/simple" }
|
326 |
dependencies = [
|
327 |
{ name = "packaging" },
|
328 |
]
|
329 |
-
sdist = { url = "https://files.pythonhosted.org/packages/
|
330 |
wheels = [
|
331 |
-
{ url = "https://files.pythonhosted.org/packages/
|
332 |
]
|
333 |
|
334 |
[[package]]
|
@@ -877,67 +847,6 @@ wheels = [
|
|
877 |
{ url = "https://files.pythonhosted.org/packages/13/a3/a812df4e2dd5696d1f351d58b8fe16a405b234ad2886a0dab9183fb78109/pycparser-2.22-py3-none-any.whl", hash = "sha256:c3702b6d3dd8c7abc1afa565d7e63d53a1d0bd86cdc24edd75470f4de499cfcc", size = 117552 },
|
878 |
]
|
879 |
|
880 |
-
[[package]]
|
881 |
-
name = "pydantic"
|
882 |
-
version = "2.9.2"
|
883 |
-
source = { registry = "https://pypi.org/simple" }
|
884 |
-
dependencies = [
|
885 |
-
{ name = "annotated-types" },
|
886 |
-
{ name = "pydantic-core" },
|
887 |
-
{ name = "typing-extensions" },
|
888 |
-
]
|
889 |
-
sdist = { url = "https://files.pythonhosted.org/packages/a9/b7/d9e3f12af310e1120c21603644a1cd86f59060e040ec5c3a80b8f05fae30/pydantic-2.9.2.tar.gz", hash = "sha256:d155cef71265d1e9807ed1c32b4c8deec042a44a50a4188b25ac67ecd81a9c0f", size = 769917 }
|
890 |
-
wheels = [
|
891 |
-
{ url = "https://files.pythonhosted.org/packages/df/e4/ba44652d562cbf0bf320e0f3810206149c8a4e99cdbf66da82e97ab53a15/pydantic-2.9.2-py3-none-any.whl", hash = "sha256:f048cec7b26778210e28a0459867920654d48e5e62db0958433636cde4254f12", size = 434928 },
|
892 |
-
]
|
893 |
-
|
894 |
-
[[package]]
|
895 |
-
name = "pydantic-core"
|
896 |
-
version = "2.23.4"
|
897 |
-
source = { registry = "https://pypi.org/simple" }
|
898 |
-
dependencies = [
|
899 |
-
{ name = "typing-extensions" },
|
900 |
-
]
|
901 |
-
sdist = { url = "https://files.pythonhosted.org/packages/e2/aa/6b6a9b9f8537b872f552ddd46dd3da230367754b6f707b8e1e963f515ea3/pydantic_core-2.23.4.tar.gz", hash = "sha256:2584f7cf844ac4d970fba483a717dbe10c1c1c96a969bf65d61ffe94df1b2863", size = 402156 }
|
902 |
-
wheels = [
|
903 |
-
{ url = "https://files.pythonhosted.org/packages/5d/30/890a583cd3f2be27ecf32b479d5d615710bb926d92da03e3f7838ff3e58b/pydantic_core-2.23.4-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:77733e3892bb0a7fa797826361ce8a9184d25c8dffaec60b7ffe928153680ba8", size = 1865160 },
|
904 |
-
{ url = "https://files.pythonhosted.org/packages/1d/9a/b634442e1253bc6889c87afe8bb59447f106ee042140bd57680b3b113ec7/pydantic_core-2.23.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1b84d168f6c48fabd1f2027a3d1bdfe62f92cade1fb273a5d68e621da0e44e6d", size = 1776777 },
|
905 |
-
{ url = "https://files.pythonhosted.org/packages/75/9a/7816295124a6b08c24c96f9ce73085032d8bcbaf7e5a781cd41aa910c891/pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df49e7a0861a8c36d089c1ed57d308623d60416dab2647a4a17fe050ba85de0e", size = 1799244 },
|
906 |
-
{ url = "https://files.pythonhosted.org/packages/a9/8f/89c1405176903e567c5f99ec53387449e62f1121894aa9fc2c4fdc51a59b/pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ff02b6d461a6de369f07ec15e465a88895f3223eb75073ffea56b84d9331f607", size = 1805307 },
|
907 |
-
{ url = "https://files.pythonhosted.org/packages/d5/a5/1a194447d0da1ef492e3470680c66048fef56fc1f1a25cafbea4bc1d1c48/pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:996a38a83508c54c78a5f41456b0103c30508fed9abcad0a59b876d7398f25fd", size = 2000663 },
|
908 |
-
{ url = "https://files.pythonhosted.org/packages/13/a5/1df8541651de4455e7d587cf556201b4f7997191e110bca3b589218745a5/pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d97683ddee4723ae8c95d1eddac7c192e8c552da0c73a925a89fa8649bf13eea", size = 2655941 },
|
909 |
-
{ url = "https://files.pythonhosted.org/packages/44/31/a3899b5ce02c4316865e390107f145089876dff7e1dfc770a231d836aed8/pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:216f9b2d7713eb98cb83c80b9c794de1f6b7e3145eef40400c62e86cee5f4e1e", size = 2052105 },
|
910 |
-
{ url = "https://files.pythonhosted.org/packages/1b/aa/98e190f8745d5ec831f6d5449344c48c0627ac5fed4e5340a44b74878f8e/pydantic_core-2.23.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6f783e0ec4803c787bcea93e13e9932edab72068f68ecffdf86a99fd5918878b", size = 1919967 },
|
911 |
-
{ url = "https://files.pythonhosted.org/packages/ae/35/b6e00b6abb2acfee3e8f85558c02a0822e9a8b2f2d812ea8b9079b118ba0/pydantic_core-2.23.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:d0776dea117cf5272382634bd2a5c1b6eb16767c223c6a5317cd3e2a757c61a0", size = 1964291 },
|
912 |
-
{ url = "https://files.pythonhosted.org/packages/13/46/7bee6d32b69191cd649bbbd2361af79c472d72cb29bb2024f0b6e350ba06/pydantic_core-2.23.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:d5f7a395a8cf1621939692dba2a6b6a830efa6b3cee787d82c7de1ad2930de64", size = 2109666 },
|
913 |
-
{ url = "https://files.pythonhosted.org/packages/39/ef/7b34f1b122a81b68ed0a7d0e564da9ccdc9a2924c8d6c6b5b11fa3a56970/pydantic_core-2.23.4-cp311-none-win32.whl", hash = "sha256:74b9127ffea03643e998e0c5ad9bd3811d3dac8c676e47db17b0ee7c3c3bf35f", size = 1732940 },
|
914 |
-
{ url = "https://files.pythonhosted.org/packages/2f/76/37b7e76c645843ff46c1d73e046207311ef298d3f7b2f7d8f6ac60113071/pydantic_core-2.23.4-cp311-none-win_amd64.whl", hash = "sha256:98d134c954828488b153d88ba1f34e14259284f256180ce659e8d83e9c05eaa3", size = 1916804 },
|
915 |
-
{ url = "https://files.pythonhosted.org/packages/74/7b/8e315f80666194b354966ec84b7d567da77ad927ed6323db4006cf915f3f/pydantic_core-2.23.4-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:f3e0da4ebaef65158d4dfd7d3678aad692f7666877df0002b8a522cdf088f231", size = 1856459 },
|
916 |
-
{ url = "https://files.pythonhosted.org/packages/14/de/866bdce10ed808323d437612aca1ec9971b981e1c52e5e42ad9b8e17a6f6/pydantic_core-2.23.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f69a8e0b033b747bb3e36a44e7732f0c99f7edd5cea723d45bc0d6e95377ffee", size = 1770007 },
|
917 |
-
{ url = "https://files.pythonhosted.org/packages/dc/69/8edd5c3cd48bb833a3f7ef9b81d7666ccddd3c9a635225214e044b6e8281/pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:723314c1d51722ab28bfcd5240d858512ffd3116449c557a1336cbe3919beb87", size = 1790245 },
|
918 |
-
{ url = "https://files.pythonhosted.org/packages/80/33/9c24334e3af796ce80d2274940aae38dd4e5676298b4398eff103a79e02d/pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bb2802e667b7051a1bebbfe93684841cc9351004e2badbd6411bf357ab8d5ac8", size = 1801260 },
|
919 |
-
{ url = "https://files.pythonhosted.org/packages/a5/6f/e9567fd90104b79b101ca9d120219644d3314962caa7948dd8b965e9f83e/pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d18ca8148bebe1b0a382a27a8ee60350091a6ddaf475fa05ef50dc35b5df6327", size = 1996872 },
|
920 |
-
{ url = "https://files.pythonhosted.org/packages/2d/ad/b5f0fe9e6cfee915dd144edbd10b6e9c9c9c9d7a56b69256d124b8ac682e/pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33e3d65a85a2a4a0dc3b092b938a4062b1a05f3a9abde65ea93b233bca0e03f2", size = 2661617 },
|
921 |
-
{ url = "https://files.pythonhosted.org/packages/06/c8/7d4b708f8d05a5cbfda3243aad468052c6e99de7d0937c9146c24d9f12e9/pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:128585782e5bfa515c590ccee4b727fb76925dd04a98864182b22e89a4e6ed36", size = 2071831 },
|
922 |
-
{ url = "https://files.pythonhosted.org/packages/89/4d/3079d00c47f22c9a9a8220db088b309ad6e600a73d7a69473e3a8e5e3ea3/pydantic_core-2.23.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:68665f4c17edcceecc112dfed5dbe6f92261fb9d6054b47d01bf6371a6196126", size = 1917453 },
|
923 |
-
{ url = "https://files.pythonhosted.org/packages/e9/88/9df5b7ce880a4703fcc2d76c8c2d8eb9f861f79d0c56f4b8f5f2607ccec8/pydantic_core-2.23.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:20152074317d9bed6b7a95ade3b7d6054845d70584216160860425f4fbd5ee9e", size = 1968793 },
|
924 |
-
{ url = "https://files.pythonhosted.org/packages/e3/b9/41f7efe80f6ce2ed3ee3c2dcfe10ab7adc1172f778cc9659509a79518c43/pydantic_core-2.23.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:9261d3ce84fa1d38ed649c3638feefeae23d32ba9182963e465d58d62203bd24", size = 2116872 },
|
925 |
-
{ url = "https://files.pythonhosted.org/packages/63/08/b59b7a92e03dd25554b0436554bf23e7c29abae7cce4b1c459cd92746811/pydantic_core-2.23.4-cp312-none-win32.whl", hash = "sha256:4ba762ed58e8d68657fc1281e9bb72e1c3e79cc5d464be146e260c541ec12d84", size = 1738535 },
|
926 |
-
{ url = "https://files.pythonhosted.org/packages/88/8d/479293e4d39ab409747926eec4329de5b7129beaedc3786eca070605d07f/pydantic_core-2.23.4-cp312-none-win_amd64.whl", hash = "sha256:97df63000f4fea395b2824da80e169731088656d1818a11b95f3b173747b6cd9", size = 1917992 },
|
927 |
-
{ url = "https://files.pythonhosted.org/packages/ad/ef/16ee2df472bf0e419b6bc68c05bf0145c49247a1095e85cee1463c6a44a1/pydantic_core-2.23.4-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:7530e201d10d7d14abce4fb54cfe5b94a0aefc87da539d0346a484ead376c3cc", size = 1856143 },
|
928 |
-
{ url = "https://files.pythonhosted.org/packages/da/fa/bc3dbb83605669a34a93308e297ab22be82dfb9dcf88c6cf4b4f264e0a42/pydantic_core-2.23.4-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:df933278128ea1cd77772673c73954e53a1c95a4fdf41eef97c2b779271bd0bd", size = 1770063 },
|
929 |
-
{ url = "https://files.pythonhosted.org/packages/4e/48/e813f3bbd257a712303ebdf55c8dc46f9589ec74b384c9f652597df3288d/pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0cb3da3fd1b6a5d0279a01877713dbda118a2a4fc6f0d821a57da2e464793f05", size = 1790013 },
|
930 |
-
{ url = "https://files.pythonhosted.org/packages/b4/e0/56eda3a37929a1d297fcab1966db8c339023bcca0b64c5a84896db3fcc5c/pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:42c6dcb030aefb668a2b7009c85b27f90e51e6a3b4d5c9bc4c57631292015b0d", size = 1801077 },
|
931 |
-
{ url = "https://files.pythonhosted.org/packages/04/be/5e49376769bfbf82486da6c5c1683b891809365c20d7c7e52792ce4c71f3/pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:696dd8d674d6ce621ab9d45b205df149399e4bb9aa34102c970b721554828510", size = 1996782 },
|
932 |
-
{ url = "https://files.pythonhosted.org/packages/bc/24/e3ee6c04f1d58cc15f37bcc62f32c7478ff55142b7b3e6d42ea374ea427c/pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2971bb5ffe72cc0f555c13e19b23c85b654dd2a8f7ab493c262071377bfce9f6", size = 2661375 },
|
933 |
-
{ url = "https://files.pythonhosted.org/packages/c1/f8/11a9006de4e89d016b8de74ebb1db727dc100608bb1e6bbe9d56a3cbbcce/pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8394d940e5d400d04cad4f75c0598665cbb81aecefaca82ca85bd28264af7f9b", size = 2071635 },
|
934 |
-
{ url = "https://files.pythonhosted.org/packages/7c/45/bdce5779b59f468bdf262a5bc9eecbae87f271c51aef628d8c073b4b4b4c/pydantic_core-2.23.4-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:0dff76e0602ca7d4cdaacc1ac4c005e0ce0dcfe095d5b5259163a80d3a10d327", size = 1916994 },
|
935 |
-
{ url = "https://files.pythonhosted.org/packages/d8/fa/c648308fe711ee1f88192cad6026ab4f925396d1293e8356de7e55be89b5/pydantic_core-2.23.4-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:7d32706badfe136888bdea71c0def994644e09fff0bfe47441deaed8e96fdbc6", size = 1968877 },
|
936 |
-
{ url = "https://files.pythonhosted.org/packages/16/16/b805c74b35607d24d37103007f899abc4880923b04929547ae68d478b7f4/pydantic_core-2.23.4-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:ed541d70698978a20eb63d8c5d72f2cc6d7079d9d90f6b50bad07826f1320f5f", size = 2116814 },
|
937 |
-
{ url = "https://files.pythonhosted.org/packages/d1/58/5305e723d9fcdf1c5a655e6a4cc2a07128bf644ff4b1d98daf7a9dbf57da/pydantic_core-2.23.4-cp313-none-win32.whl", hash = "sha256:3d5639516376dce1940ea36edf408c554475369f5da2abd45d44621cb616f769", size = 1738360 },
|
938 |
-
{ url = "https://files.pythonhosted.org/packages/a5/ae/e14b0ff8b3f48e02394d8acd911376b7b66e164535687ef7dc24ea03072f/pydantic_core-2.23.4-cp313-none-win_amd64.whl", hash = "sha256:5a1504ad17ba4210df3a045132a7baeeba5a200e930f57512ee02909fc5c4cb5", size = 1919411 },
|
939 |
-
]
|
940 |
-
|
941 |
[[package]]
|
942 |
name = "pyee"
|
943 |
version = "12.0.0"
|
@@ -959,44 +868,6 @@ wheels = [
|
|
959 |
{ url = "https://files.pythonhosted.org/packages/f7/3f/01c8b82017c199075f8f788d0d906b9ffbbc5a47dc9918a945e13d5a2bda/pygments-2.18.0-py3-none-any.whl", hash = "sha256:b8e6aca0523f3ab76fee51799c488e38782ac06eafcf95e7ba832985c8e7b13a", size = 1205513 },
|
960 |
]
|
961 |
|
962 |
-
[[package]]
|
963 |
-
name = "pyinstrument"
|
964 |
-
version = "4.7.3"
|
965 |
-
source = { registry = "https://pypi.org/simple" }
|
966 |
-
sdist = { url = "https://files.pythonhosted.org/packages/df/ff/a899b0cb585f88a515703181b02488ba83c82a0df886d8eacb6921da6642/pyinstrument-4.7.3.tar.gz", hash = "sha256:3ad61041ff1880d4c99d3384cd267e38a0a6472b5a4dd765992db376bd4394c8", size = 129732 }
|
967 |
-
wheels = [
|
968 |
-
{ url = "https://files.pythonhosted.org/packages/96/f1/84bfbcf6dab75c93d1b9e7437ef22305a13ba0f96060af454a80a05d0f89/pyinstrument-4.7.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:77594adf4713bc3e430e300561a2d837213cf9015414c0e0de6aef0cb9cebd80", size = 111052 },
|
969 |
-
{ url = "https://files.pythonhosted.org/packages/69/58/21d86b4d545149a9a9ef3bd3166917b9a329f5f4521632fdbe047ab45caf/pyinstrument-4.7.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:70afa765c06e4f7605033b85ef82ed946ec8e6ae1835e25f6cbb01205a624197", size = 103048 },
|
970 |
-
{ url = "https://files.pythonhosted.org/packages/bb/90/dde09a02620335cf6969823022904293f2c2288dd436aa4b10dd3d26738b/pyinstrument-4.7.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7b1321514863be18138a6d761696b3f6e8645390dd2f6c8a6d66a453f0d5187c", size = 125390 },
|
971 |
-
{ url = "https://files.pythonhosted.org/packages/cd/32/884be2d9b1c27ffeaceb1f8a4d36b9cb97f2ddd40da4af487d254acfdb1f/pyinstrument-4.7.3-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:de40b44ff2fe78493b944b679cc084e72b2648c37a96fcfbccb9171a4449e509", size = 124324 },
|
972 |
-
{ url = "https://files.pythonhosted.org/packages/f2/f6/3359e7b452955c36e285c24c357f784f2d963cd5247326c0435cd2917ade/pyinstrument-4.7.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2a7c481daec4bd77a3dbfbe01a0155e03352dd700f3c3efe4bdbc30821b20e19", size = 125463 },
|
973 |
-
{ url = "https://files.pythonhosted.org/packages/64/5c/12f4b9aa37057ba832b81a0039abf75a0551ae75bf843efb6190becaca89/pyinstrument-4.7.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:ae2c966c91da630a23dbff5f7e61ad2eee133cfaf1e4acf7e09fcf506cbb6251", size = 125458 },
|
974 |
-
{ url = "https://files.pythonhosted.org/packages/d4/10/2bbd12bea2ab8bf6fdd06171710b3873776501abfb12d27750ccdc9d62df/pyinstrument-4.7.3-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:fa2715e3ac3ce2f4b9c4e468a9a4faf43ca645beea002cb47533902576f4f64d", size = 124969 },
|
975 |
-
{ url = "https://files.pythonhosted.org/packages/18/f3/de97a107634cccadc93b0e9b2adb577dc22b785b0f1dbbfbe63921dbf81b/pyinstrument-4.7.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:61db15f8b59a3a1964041a8df260667fb5dabddd928301e3580cf93d7a05e352", size = 125205 },
|
976 |
-
{ url = "https://files.pythonhosted.org/packages/aa/a0/da8fcc394074aacf2050d6ff84ba87cb2b508e70eb0d482456bd7edffe7a/pyinstrument-4.7.3-cp311-cp311-win32.whl", hash = "sha256:4766bbb2b451460432c97baf00bbda56653429671e8daec344d343f21fb05b8f", size = 104612 },
|
977 |
-
{ url = "https://files.pythonhosted.org/packages/85/9c/e745669202474f513350dfaf18c0f886d857464fd5301853f1841f25108a/pyinstrument-4.7.3-cp311-cp311-win_amd64.whl", hash = "sha256:b2d2a0e401db6800f63de0539415cdff46b138914d771a46db0b3f673f9827e7", size = 105417 },
|
978 |
-
{ url = "https://files.pythonhosted.org/packages/7f/b5/f8ccdad3fcb8a722a99c236a35eb32fb49962a74992a9c5086207c86491f/pyinstrument-4.7.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:7c29f7a23e0f704f5f21aeeb47193460601e7359d09156ea043395870494b39a", size = 111083 },
|
979 |
-
{ url = "https://files.pythonhosted.org/packages/c0/4d/87cf8287127f39274b979e11b7fc3a7520a433c03000db4f0e3c63da891f/pyinstrument-4.7.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:84ceb25f24ceb03dc770b6c142ec4419506d3a04d66d778810cb8da76df25651", size = 103110 },
|
980 |
-
{ url = "https://files.pythonhosted.org/packages/d6/fd/3edb6129d91be4d08af87a47091c52a114949305d710d6bc223e89226957/pyinstrument-4.7.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d564d6f6151d3cab28430092cdcbd4aefe0834551af4b4f97e6e57025a348557", size = 126576 },
|
981 |
-
{ url = "https://files.pythonhosted.org/packages/bd/7c/42271ae6c70f22c57ff56239c9695ed5e4b34690767da0fce997c8b9b662/pyinstrument-4.7.3-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7e23ce5fcc30346e576b98ca24bd2a9a68cbc42b90cdb0d8f376fa82cee2fe23", size = 125508 },
|
982 |
-
{ url = "https://files.pythonhosted.org/packages/40/e7/67f74f224660fca73b132a178870003342b626fa9579f8a2fd77f2f5736f/pyinstrument-4.7.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e23d5ad174d2a488c164abee4407f3f3a6e6d5721ab1fab9e0ad9570631704c2", size = 126868 },
|
983 |
-
{ url = "https://files.pythonhosted.org/packages/53/eb/f779ab14b87e2163aec08eb1a2d6da66c72c2bf0abc910d4b0070b90710b/pyinstrument-4.7.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:d87749f68b9cc221628aab989a4a73b16030c27c714ecd83892d716f863d9739", size = 126557 },
|
984 |
-
{ url = "https://files.pythonhosted.org/packages/96/f8/eb5366862a3523b59fc4c2a3b4d248d2c1e36dbbbd6f7a6b43ea3e0a4f45/pyinstrument-4.7.3-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:897d09c876f18b713498be21430b39428a9254ffec0c6c06796fce0e6a8fe437", size = 126301 },
|
985 |
-
{ url = "https://files.pythonhosted.org/packages/97/44/b234e50b35cda4f8405949f2d70d780ef2ce057a147ac45848997830116e/pyinstrument-4.7.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:2092910e745cfd0a62dadf041afb38239195244871ee127b1028e7e790602e6b", size = 126737 },
|
986 |
-
{ url = "https://files.pythonhosted.org/packages/94/b5/38b600f8cc25d335738cd3e08cac74a328fa3525c03ff61b9a38050336c1/pyinstrument-4.7.3-cp312-cp312-win32.whl", hash = "sha256:e9824e11290f6f2772c257cc0bd07f59405759287db6ebcbb06f962a3eba68fb", size = 104708 },
|
987 |
-
{ url = "https://files.pythonhosted.org/packages/32/74/95b1773d7c8c405813b820985021db68df83e624a9cc9640fc06c9bbdd59/pyinstrument-4.7.3-cp312-cp312-win_amd64.whl", hash = "sha256:cf1e67b37e936f647ce731fff5d2f54e102813274d350671dc5961ec8b46b3ff", size = 105496 },
|
988 |
-
{ url = "https://files.pythonhosted.org/packages/b6/96/5b9102380458702ce7ce5d09c625a25dc5d42cb243a89efb8273029ae6dc/pyinstrument-4.7.3-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:6de792dc65dcc75e73b721f4e89aa60a4d2f8617e5a5da060244058018ad0399", size = 110998 },
|
989 |
-
{ url = "https://files.pythonhosted.org/packages/45/33/97aa76dbb34af7d54aa007753615c0f1b5af876db7ce41ca7a6f51e4033a/pyinstrument-4.7.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:73da379506a09cdff2fdd23a0b3eb8f020f473d019f604538e0e5045613e33d4", size = 103026 },
|
990 |
-
{ url = "https://files.pythonhosted.org/packages/97/39/04c6d79e8af80197125a5fc2c8485ce823e0f28aecd4302d55df0225b011/pyinstrument-4.7.3-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:21e05f53810a6ff5fa261da838935fd1b2ab2bf30a7c053f6c72bcaaa6de0933", size = 126690 },
|
991 |
-
{ url = "https://files.pythonhosted.org/packages/5c/3c/e23cac0427bf422eba75e2157b3daea02a16d68eaad384f9c11dc4eda675/pyinstrument-4.7.3-cp313-cp313-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d648596ea04409ca3ca260029041ed7fa046b776205bf9a0b75cda0a4f4d2515", size = 125596 },
|
992 |
-
{ url = "https://files.pythonhosted.org/packages/76/a8/14eeb9e33698fc47160210b225d0d2dff6b0d2fd21b0e59e378302c89b55/pyinstrument-4.7.3-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3d98997347047a217ef6b844273d3753e543e0984f2220e9dd284cbef6054c2a", size = 127001 },
|
993 |
-
{ url = "https://files.pythonhosted.org/packages/5f/4d/1d3d03fff607b145b647c46ebe25d61adc689d0359378fc05181d4007287/pyinstrument-4.7.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:7f09ebad95af94f5427c20005fc7ba84a0a3deae6324434d7ec3be99d369bf37", size = 126696 },
|
994 |
-
{ url = "https://files.pythonhosted.org/packages/f9/b0/d80246d8dc2020f08dd836a483c1d7e7159d2abc6e5215c2908108562380/pyinstrument-4.7.3-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:8a66aee3d2cf0cc6b8e57cb189fd9fb16d13b8d538419999596ce4f58b5d4a9a", size = 126467 },
|
995 |
-
{ url = "https://files.pythonhosted.org/packages/d9/a8/d87e440011cda09cb219feb5f0e848ad14635a26a962fe601b277e7ddd89/pyinstrument-4.7.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:eaa45270af0b9d86f1cef705520e9b43f4a1cd18397083f8a594a28f898d078b", size = 126880 },
|
996 |
-
{ url = "https://files.pythonhosted.org/packages/48/00/aece845913ac703f6575dcb0c29f7a651a6fd11b922025b9395d470c04e1/pyinstrument-4.7.3-cp313-cp313-win32.whl", hash = "sha256:6e85b34a9b8ed4df4deaa0afe63bc765ea29003eb5b9b3bc0323f7ad7f7cd0fd", size = 104713 },
|
997 |
-
{ url = "https://files.pythonhosted.org/packages/21/81/1f8dcb1ad94bafc777dc2e6fd8fa82dc838ccaf01d9051f019ae1ff1641c/pyinstrument-4.7.3-cp313-cp313-win_amd64.whl", hash = "sha256:6002ea1018d6d6f9b6f1c66b3e14805213573bd69f79b2e7ad2c507441b3e73e", size = 105501 },
|
998 |
-
]
|
999 |
-
|
1000 |
[[package]]
|
1001 |
name = "pytest"
|
1002 |
version = "8.3.3"
|
@@ -1076,7 +947,7 @@ wheels = [
|
|
1076 |
|
1077 |
[[package]]
|
1078 |
name = "python-fasthtml"
|
1079 |
-
version = "0.6.
|
1080 |
source = { registry = "https://pypi.org/simple" }
|
1081 |
dependencies = [
|
1082 |
{ name = "beautifulsoup4" },
|
@@ -1090,9 +961,9 @@ dependencies = [
|
|
1090 |
{ name = "starlette" },
|
1091 |
{ name = "uvicorn", extra = ["standard"] },
|
1092 |
]
|
1093 |
-
sdist = { url = "https://files.pythonhosted.org/packages/
|
1094 |
wheels = [
|
1095 |
-
{ url = "https://files.pythonhosted.org/packages/
|
1096 |
]
|
1097 |
|
1098 |
[[package]]
|
@@ -1415,7 +1286,6 @@ dependencies = [
|
|
1415 |
|
1416 |
[package.dev-dependencies]
|
1417 |
dev = [
|
1418 |
-
{ name = "devicorn" },
|
1419 |
{ name = "ipykernel" },
|
1420 |
{ name = "lxml" },
|
1421 |
{ name = "pandas" },
|
@@ -1433,7 +1303,6 @@ requires-dist = [
|
|
1433 |
|
1434 |
[package.metadata.requires-dev]
|
1435 |
dev = [
|
1436 |
-
{ name = "devicorn", editable = "../devicorn" },
|
1437 |
{ name = "ipykernel", specifier = ">=6.29.5" },
|
1438 |
{ name = "lxml", specifier = ">=5.3.0" },
|
1439 |
{ name = "pandas", specifier = ">=2.2.2" },
|
|
|
2 |
requires-python = ">=3.11"
|
3 |
resolution-markers = [
|
4 |
"python_full_version < '3.12'",
|
5 |
+
"python_full_version >= '3.12'",
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
6 |
]
|
7 |
|
8 |
[[package]]
|
|
|
107 |
|
108 |
[[package]]
|
109 |
name = "charset-normalizer"
|
110 |
+
version = "3.4.0"
|
111 |
+
source = { registry = "https://pypi.org/simple" }
|
112 |
+
sdist = { url = "https://files.pythonhosted.org/packages/f2/4f/e1808dc01273379acc506d18f1504eb2d299bd4131743b9fc54d7be4df1e/charset_normalizer-3.4.0.tar.gz", hash = "sha256:223217c3d4f82c3ac5e29032b3f1c2eb0fb591b72161f86d93f5719079dae93e", size = 106620 }
|
113 |
+
wheels = [
|
114 |
+
{ url = "https://files.pythonhosted.org/packages/9c/61/73589dcc7a719582bf56aae309b6103d2762b526bffe189d635a7fcfd998/charset_normalizer-3.4.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:0d99dd8ff461990f12d6e42c7347fd9ab2532fb70e9621ba520f9e8637161d7c", size = 193339 },
|
115 |
+
{ url = "https://files.pythonhosted.org/packages/77/d5/8c982d58144de49f59571f940e329ad6e8615e1e82ef84584c5eeb5e1d72/charset_normalizer-3.4.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c57516e58fd17d03ebe67e181a4e4e2ccab1168f8c2976c6a334d4f819fe5944", size = 124366 },
|
116 |
+
{ url = "https://files.pythonhosted.org/packages/bf/19/411a64f01ee971bed3231111b69eb56f9331a769072de479eae7de52296d/charset_normalizer-3.4.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:6dba5d19c4dfab08e58d5b36304b3f92f3bd5d42c1a3fa37b5ba5cdf6dfcbcee", size = 118874 },
|
117 |
+
{ url = "https://files.pythonhosted.org/packages/4c/92/97509850f0d00e9f14a46bc751daabd0ad7765cff29cdfb66c68b6dad57f/charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf4475b82be41b07cc5e5ff94810e6a01f276e37c2d55571e3fe175e467a1a1c", size = 138243 },
|
118 |
+
{ url = "https://files.pythonhosted.org/packages/e2/29/d227805bff72ed6d6cb1ce08eec707f7cfbd9868044893617eb331f16295/charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ce031db0408e487fd2775d745ce30a7cd2923667cf3b69d48d219f1d8f5ddeb6", size = 148676 },
|
119 |
+
{ url = "https://files.pythonhosted.org/packages/13/bc/87c2c9f2c144bedfa62f894c3007cd4530ba4b5351acb10dc786428a50f0/charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8ff4e7cdfdb1ab5698e675ca622e72d58a6fa2a8aa58195de0c0061288e6e3ea", size = 141289 },
|
120 |
+
{ url = "https://files.pythonhosted.org/packages/eb/5b/6f10bad0f6461fa272bfbbdf5d0023b5fb9bc6217c92bf068fa5a99820f5/charset_normalizer-3.4.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3710a9751938947e6327ea9f3ea6332a09bf0ba0c09cae9cb1f250bd1f1549bc", size = 142585 },
|
121 |
+
{ url = "https://files.pythonhosted.org/packages/3b/a0/a68980ab8a1f45a36d9745d35049c1af57d27255eff8c907e3add84cf68f/charset_normalizer-3.4.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:82357d85de703176b5587dbe6ade8ff67f9f69a41c0733cf2425378b49954de5", size = 144408 },
|
122 |
+
{ url = "https://files.pythonhosted.org/packages/d7/a1/493919799446464ed0299c8eef3c3fad0daf1c3cd48bff9263c731b0d9e2/charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:47334db71978b23ebcf3c0f9f5ee98b8d65992b65c9c4f2d34c2eaf5bcaf0594", size = 139076 },
|
123 |
+
{ url = "https://files.pythonhosted.org/packages/fb/9d/9c13753a5a6e0db4a0a6edb1cef7aee39859177b64e1a1e748a6e3ba62c2/charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:8ce7fd6767a1cc5a92a639b391891bf1c268b03ec7e021c7d6d902285259685c", size = 146874 },
|
124 |
+
{ url = "https://files.pythonhosted.org/packages/75/d2/0ab54463d3410709c09266dfb416d032a08f97fd7d60e94b8c6ef54ae14b/charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:f1a2f519ae173b5b6a2c9d5fa3116ce16e48b3462c8b96dfdded11055e3d6365", size = 150871 },
|
125 |
+
{ url = "https://files.pythonhosted.org/packages/8d/c9/27e41d481557be53d51e60750b85aa40eaf52b841946b3cdeff363105737/charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:63bc5c4ae26e4bc6be6469943b8253c0fd4e4186c43ad46e713ea61a0ba49129", size = 148546 },
|
126 |
+
{ url = "https://files.pythonhosted.org/packages/ee/44/4f62042ca8cdc0cabf87c0fc00ae27cd8b53ab68be3605ba6d071f742ad3/charset_normalizer-3.4.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:bcb4f8ea87d03bc51ad04add8ceaf9b0f085ac045ab4d74e73bbc2dc033f0236", size = 143048 },
|
127 |
+
{ url = "https://files.pythonhosted.org/packages/01/f8/38842422988b795220eb8038745d27a675ce066e2ada79516c118f291f07/charset_normalizer-3.4.0-cp311-cp311-win32.whl", hash = "sha256:9ae4ef0b3f6b41bad6366fb0ea4fc1d7ed051528e113a60fa2a65a9abb5b1d99", size = 94389 },
|
128 |
+
{ url = "https://files.pythonhosted.org/packages/0b/6e/b13bd47fa9023b3699e94abf565b5a2f0b0be6e9ddac9812182596ee62e4/charset_normalizer-3.4.0-cp311-cp311-win_amd64.whl", hash = "sha256:cee4373f4d3ad28f1ab6290684d8e2ebdb9e7a1b74fdc39e4c211995f77bec27", size = 101752 },
|
129 |
+
{ url = "https://files.pythonhosted.org/packages/d3/0b/4b7a70987abf9b8196845806198975b6aab4ce016632f817ad758a5aa056/charset_normalizer-3.4.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0713f3adb9d03d49d365b70b84775d0a0d18e4ab08d12bc46baa6132ba78aaf6", size = 194445 },
|
130 |
+
{ url = "https://files.pythonhosted.org/packages/50/89/354cc56cf4dd2449715bc9a0f54f3aef3dc700d2d62d1fa5bbea53b13426/charset_normalizer-3.4.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:de7376c29d95d6719048c194a9cf1a1b0393fbe8488a22008610b0361d834ecf", size = 125275 },
|
131 |
+
{ url = "https://files.pythonhosted.org/packages/fa/44/b730e2a2580110ced837ac083d8ad222343c96bb6b66e9e4e706e4d0b6df/charset_normalizer-3.4.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:4a51b48f42d9358460b78725283f04bddaf44a9358197b889657deba38f329db", size = 119020 },
|
132 |
+
{ url = "https://files.pythonhosted.org/packages/9d/e4/9263b8240ed9472a2ae7ddc3e516e71ef46617fe40eaa51221ccd4ad9a27/charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b295729485b06c1a0683af02a9e42d2caa9db04a373dc38a6a58cdd1e8abddf1", size = 139128 },
|
133 |
+
{ url = "https://files.pythonhosted.org/packages/6b/e3/9f73e779315a54334240353eaea75854a9a690f3f580e4bd85d977cb2204/charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ee803480535c44e7f5ad00788526da7d85525cfefaf8acf8ab9a310000be4b03", size = 149277 },
|
134 |
+
{ url = "https://files.pythonhosted.org/packages/1a/cf/f1f50c2f295312edb8a548d3fa56a5c923b146cd3f24114d5adb7e7be558/charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3d59d125ffbd6d552765510e3f31ed75ebac2c7470c7274195b9161a32350284", size = 142174 },
|
135 |
+
{ url = "https://files.pythonhosted.org/packages/16/92/92a76dc2ff3a12e69ba94e7e05168d37d0345fa08c87e1fe24d0c2a42223/charset_normalizer-3.4.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8cda06946eac330cbe6598f77bb54e690b4ca93f593dee1568ad22b04f347c15", size = 143838 },
|
136 |
+
{ url = "https://files.pythonhosted.org/packages/a4/01/2117ff2b1dfc61695daf2babe4a874bca328489afa85952440b59819e9d7/charset_normalizer-3.4.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:07afec21bbbbf8a5cc3651aa96b980afe2526e7f048fdfb7f1014d84acc8b6d8", size = 146149 },
|
137 |
+
{ url = "https://files.pythonhosted.org/packages/f6/9b/93a332b8d25b347f6839ca0a61b7f0287b0930216994e8bf67a75d050255/charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6b40e8d38afe634559e398cc32b1472f376a4099c75fe6299ae607e404c033b2", size = 140043 },
|
138 |
+
{ url = "https://files.pythonhosted.org/packages/ab/f6/7ac4a01adcdecbc7a7587767c776d53d369b8b971382b91211489535acf0/charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:b8dcd239c743aa2f9c22ce674a145e0a25cb1566c495928440a181ca1ccf6719", size = 148229 },
|
139 |
+
{ url = "https://files.pythonhosted.org/packages/9d/be/5708ad18161dee7dc6a0f7e6cf3a88ea6279c3e8484844c0590e50e803ef/charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:84450ba661fb96e9fd67629b93d2941c871ca86fc38d835d19d4225ff946a631", size = 151556 },
|
140 |
+
{ url = "https://files.pythonhosted.org/packages/5a/bb/3d8bc22bacb9eb89785e83e6723f9888265f3a0de3b9ce724d66bd49884e/charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:44aeb140295a2f0659e113b31cfe92c9061622cadbc9e2a2f7b8ef6b1e29ef4b", size = 149772 },
|
141 |
+
{ url = "https://files.pythonhosted.org/packages/f7/fa/d3fc622de05a86f30beea5fc4e9ac46aead4731e73fd9055496732bcc0a4/charset_normalizer-3.4.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:1db4e7fefefd0f548d73e2e2e041f9df5c59e178b4c72fbac4cc6f535cfb1565", size = 144800 },
|
142 |
+
{ url = "https://files.pythonhosted.org/packages/9a/65/bdb9bc496d7d190d725e96816e20e2ae3a6fa42a5cac99c3c3d6ff884118/charset_normalizer-3.4.0-cp312-cp312-win32.whl", hash = "sha256:5726cf76c982532c1863fb64d8c6dd0e4c90b6ece9feb06c9f202417a31f7dd7", size = 94836 },
|
143 |
+
{ url = "https://files.pythonhosted.org/packages/3e/67/7b72b69d25b89c0b3cea583ee372c43aa24df15f0e0f8d3982c57804984b/charset_normalizer-3.4.0-cp312-cp312-win_amd64.whl", hash = "sha256:b197e7094f232959f8f20541ead1d9862ac5ebea1d58e9849c1bf979255dfac9", size = 102187 },
|
144 |
+
{ url = "https://files.pythonhosted.org/packages/f3/89/68a4c86f1a0002810a27f12e9a7b22feb198c59b2f05231349fbce5c06f4/charset_normalizer-3.4.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:dd4eda173a9fcccb5f2e2bd2a9f423d180194b1bf17cf59e3269899235b2a114", size = 194617 },
|
145 |
+
{ url = "https://files.pythonhosted.org/packages/4f/cd/8947fe425e2ab0aa57aceb7807af13a0e4162cd21eee42ef5b053447edf5/charset_normalizer-3.4.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e9e3c4c9e1ed40ea53acf11e2a386383c3304212c965773704e4603d589343ed", size = 125310 },
|
146 |
+
{ url = "https://files.pythonhosted.org/packages/5b/f0/b5263e8668a4ee9becc2b451ed909e9c27058337fda5b8c49588183c267a/charset_normalizer-3.4.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:92a7e36b000bf022ef3dbb9c46bfe2d52c047d5e3f3343f43204263c5addc250", size = 119126 },
|
147 |
+
{ url = "https://files.pythonhosted.org/packages/ff/6e/e445afe4f7fda27a533f3234b627b3e515a1b9429bc981c9a5e2aa5d97b6/charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:54b6a92d009cbe2fb11054ba694bc9e284dad30a26757b1e372a1fdddaf21920", size = 139342 },
|
148 |
+
{ url = "https://files.pythonhosted.org/packages/a1/b2/4af9993b532d93270538ad4926c8e37dc29f2111c36f9c629840c57cd9b3/charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ffd9493de4c922f2a38c2bf62b831dcec90ac673ed1ca182fe11b4d8e9f2a64", size = 149383 },
|
149 |
+
{ url = "https://files.pythonhosted.org/packages/fb/6f/4e78c3b97686b871db9be6f31d64e9264e889f8c9d7ab33c771f847f79b7/charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:35c404d74c2926d0287fbd63ed5d27eb911eb9e4a3bb2c6d294f3cfd4a9e0c23", size = 142214 },
|
150 |
+
{ url = "https://files.pythonhosted.org/packages/2b/c9/1c8fe3ce05d30c87eff498592c89015b19fade13df42850aafae09e94f35/charset_normalizer-3.4.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4796efc4faf6b53a18e3d46343535caed491776a22af773f366534056c4e1fbc", size = 144104 },
|
151 |
+
{ url = "https://files.pythonhosted.org/packages/ee/68/efad5dcb306bf37db7db338338e7bb8ebd8cf38ee5bbd5ceaaaa46f257e6/charset_normalizer-3.4.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e7fdd52961feb4c96507aa649550ec2a0d527c086d284749b2f582f2d40a2e0d", size = 146255 },
|
152 |
+
{ url = "https://files.pythonhosted.org/packages/0c/75/1ed813c3ffd200b1f3e71121c95da3f79e6d2a96120163443b3ad1057505/charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:92db3c28b5b2a273346bebb24857fda45601aef6ae1c011c0a997106581e8a88", size = 140251 },
|
153 |
+
{ url = "https://files.pythonhosted.org/packages/7d/0d/6f32255c1979653b448d3c709583557a4d24ff97ac4f3a5be156b2e6a210/charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:ab973df98fc99ab39080bfb0eb3a925181454d7c3ac8a1e695fddfae696d9e90", size = 148474 },
|
154 |
+
{ url = "https://files.pythonhosted.org/packages/ac/a0/c1b5298de4670d997101fef95b97ac440e8c8d8b4efa5a4d1ef44af82f0d/charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:4b67fdab07fdd3c10bb21edab3cbfe8cf5696f453afce75d815d9d7223fbe88b", size = 151849 },
|
155 |
+
{ url = "https://files.pythonhosted.org/packages/04/4f/b3961ba0c664989ba63e30595a3ed0875d6790ff26671e2aae2fdc28a399/charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:aa41e526a5d4a9dfcfbab0716c7e8a1b215abd3f3df5a45cf18a12721d31cb5d", size = 149781 },
|
156 |
+
{ url = "https://files.pythonhosted.org/packages/d8/90/6af4cd042066a4adad58ae25648a12c09c879efa4849c705719ba1b23d8c/charset_normalizer-3.4.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ffc519621dce0c767e96b9c53f09c5d215578e10b02c285809f76509a3931482", size = 144970 },
|
157 |
+
{ url = "https://files.pythonhosted.org/packages/cc/67/e5e7e0cbfefc4ca79025238b43cdf8a2037854195b37d6417f3d0895c4c2/charset_normalizer-3.4.0-cp313-cp313-win32.whl", hash = "sha256:f19c1585933c82098c2a520f8ec1227f20e339e33aca8fa6f956f6691b784e67", size = 94973 },
|
158 |
+
{ url = "https://files.pythonhosted.org/packages/65/97/fc9bbc54ee13d33dc54a7fcf17b26368b18505500fc01e228c27b5222d80/charset_normalizer-3.4.0-cp313-cp313-win_amd64.whl", hash = "sha256:707b82d19e65c9bd28b81dde95249b07bf9f5b90ebe1ef17d9b57473f8a64b7b", size = 102308 },
|
159 |
+
{ url = "https://files.pythonhosted.org/packages/bf/9b/08c0432272d77b04803958a4598a51e2a4b51c06640af8b8f0f908c18bf2/charset_normalizer-3.4.0-py3-none-any.whl", hash = "sha256:fe9f97feb71aa9896b81973a7bbada8c49501dc73e58a10fcef6663af95e5079", size = 49446 },
|
160 |
]
|
161 |
|
162 |
[[package]]
|
|
|
194 |
|
195 |
[[package]]
|
196 |
name = "coverage"
|
197 |
+
version = "7.6.2"
|
198 |
+
source = { registry = "https://pypi.org/simple" }
|
199 |
+
sdist = { url = "https://files.pythonhosted.org/packages/9a/60/e781e8302e7b28f21ce06e30af077f856aa2cb4cf2253287dae9a593d509/coverage-7.6.2.tar.gz", hash = "sha256:a5f81e68aa62bc0cfca04f7b19eaa8f9c826b53fc82ab9e2121976dc74f131f3", size = 797872 }
|
200 |
+
wheels = [
|
201 |
+
{ url = "https://files.pythonhosted.org/packages/a5/29/72da824da4182f518b054c21552b7ed2473a4e4c6ac616298209808a1a5c/coverage-7.6.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:bb21bac7783c1bf6f4bbe68b1e0ff0d20e7e7732cfb7995bc8d96e23aa90fc7b", size = 206667 },
|
202 |
+
{ url = "https://files.pythonhosted.org/packages/23/52/c15dcf3cf575256c7c0992e441cd41092a6c519d65abe1eb5567aab3d8e8/coverage-7.6.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a7b2e437fbd8fae5bc7716b9c7ff97aecc95f0b4d56e4ca08b3c8d8adcaadb84", size = 207111 },
|
203 |
+
{ url = "https://files.pythonhosted.org/packages/92/61/0d46dc26cf9f711b7b6078a54680665a5c2d62ec15991adb51e79236c699/coverage-7.6.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:536f77f2bf5797983652d1d55f1a7272a29afcc89e3ae51caa99b2db4e89d658", size = 239050 },
|
204 |
+
{ url = "https://files.pythonhosted.org/packages/3b/cb/9de71bade0343a0793f645f78a0e409248d85a2e5b4c4a9a1697c3b2e3d2/coverage-7.6.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f361296ca7054f0936b02525646b2731b32c8074ba6defab524b79b2b7eeac72", size = 236454 },
|
205 |
+
{ url = "https://files.pythonhosted.org/packages/f2/81/b0dc02487447c4a56cf2eed5c57735097f77aeff582277a35f1f70713a8d/coverage-7.6.2-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7926d8d034e06b479797c199747dd774d5e86179f2ce44294423327a88d66ca7", size = 238320 },
|
206 |
+
{ url = "https://files.pythonhosted.org/packages/60/90/76815a76234050a87d0d1438a34820c1b857dd17353855c02bddabbedea8/coverage-7.6.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:0bbae11c138585c89fb4e991faefb174a80112e1a7557d507aaa07675c62e66b", size = 237250 },
|
207 |
+
{ url = "https://files.pythonhosted.org/packages/f6/bd/760a599c08c882d97382855264586bba2604901029c3f6bec5710477ae81/coverage-7.6.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:fcad7d5d2bbfeae1026b395036a8aa5abf67e8038ae7e6a25c7d0f88b10a8e6a", size = 235880 },
|
208 |
+
{ url = "https://files.pythonhosted.org/packages/83/de/41c3b90a779e473ae1ca325542aa5fa5464b7d2061288e9c22ba5f1deaa3/coverage-7.6.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f01e53575f27097d75d42de33b1b289c74b16891ce576d767ad8c48d17aeb5e0", size = 236653 },
|
209 |
+
{ url = "https://files.pythonhosted.org/packages/f4/90/61fe2721b9a9d9446e6c3ca33b6569e81d2a9a795ddfe786a66bf54035b7/coverage-7.6.2-cp311-cp311-win32.whl", hash = "sha256:7781f4f70c9b0b39e1b129b10c7d43a4e0c91f90c60435e6da8288efc2b73438", size = 209251 },
|
210 |
+
{ url = "https://files.pythonhosted.org/packages/96/87/d586f2b12b98288fc874d366cd8d5601f5a374cb75853647a3e4d02e4eb0/coverage-7.6.2-cp311-cp311-win_amd64.whl", hash = "sha256:9bcd51eeca35a80e76dc5794a9dd7cb04b97f0e8af620d54711793bfc1fbba4b", size = 210083 },
|
211 |
+
{ url = "https://files.pythonhosted.org/packages/3f/ac/1cca5ed5cf512a71cdd6e3afb75a5ef196f7ef9772be9192dadaaa5cfc1c/coverage-7.6.2-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:ebc94fadbd4a3f4215993326a6a00e47d79889391f5659bf310f55fe5d9f581c", size = 206856 },
|
212 |
+
{ url = "https://files.pythonhosted.org/packages/e4/58/030354d250f107a95e7aca24c7fd238709a3c7df3083cb206368798e637a/coverage-7.6.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:9681516288e3dcf0aa7c26231178cc0be6cac9705cac06709f2353c5b406cfea", size = 207098 },
|
213 |
+
{ url = "https://files.pythonhosted.org/packages/03/df/5f2cd6048d44a54bb5f58f8ece4efbc5b686ed49f8bd8dbf41eb2a6a687f/coverage-7.6.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8d9c5d13927d77af4fbe453953810db766f75401e764727e73a6ee4f82527b3e", size = 240109 },
|
214 |
+
{ url = "https://files.pythonhosted.org/packages/d3/18/7c53887643d921faa95529643b1b33e60ebba30ab835c8b5abd4e54d946b/coverage-7.6.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b92f9ca04b3e719d69b02dc4a69debb795af84cb7afd09c5eb5d54b4a1ae2191", size = 237141 },
|
215 |
+
{ url = "https://files.pythonhosted.org/packages/d2/79/339bdf597d128374e6150c089b37436ba694585d769cabf6d5abd73a1365/coverage-7.6.2-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0ff2ef83d6d0b527b5c9dad73819b24a2f76fdddcfd6c4e7a4d7e73ecb0656b4", size = 239210 },
|
216 |
+
{ url = "https://files.pythonhosted.org/packages/a9/62/7310c6de2bcb8a42f91094d41f0d4793ccda5a54621be3db76a156556cf2/coverage-7.6.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:47ccb6e99a3031ffbbd6e7cc041e70770b4fe405370c66a54dbf26a500ded80b", size = 238698 },
|
217 |
+
{ url = "https://files.pythonhosted.org/packages/f2/cb/ccb23c084d7f581f770dc7ed547dc5b50763334ad6ce26087a9ad0b5b26d/coverage-7.6.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:a867d26f06bcd047ef716175b2696b315cb7571ccb951006d61ca80bbc356e9e", size = 237000 },
|
218 |
+
{ url = "https://files.pythonhosted.org/packages/e7/ab/58de9e2f94e4dc91b84d6e2705aa1e9d5447a2669fe113b4bbce6d2224a1/coverage-7.6.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:cdfcf2e914e2ba653101157458afd0ad92a16731eeba9a611b5cbb3e7124e74b", size = 238666 },
|
219 |
+
{ url = "https://files.pythonhosted.org/packages/6c/dc/8be87b9ed5dbd4892b603f41088b41982768e928734e5bdce67d2ddd460a/coverage-7.6.2-cp312-cp312-win32.whl", hash = "sha256:f9035695dadfb397bee9eeaf1dc7fbeda483bf7664a7397a629846800ce6e276", size = 209489 },
|
220 |
+
{ url = "https://files.pythonhosted.org/packages/64/3a/3f44e55273a58bfb39b87ad76541bbb81d14de916b034fdb39971cc99ffe/coverage-7.6.2-cp312-cp312-win_amd64.whl", hash = "sha256:5ed69befa9a9fc796fe015a7040c9398722d6b97df73a6b608e9e275fa0932b0", size = 210270 },
|
221 |
+
{ url = "https://files.pythonhosted.org/packages/ae/99/c9676a75b57438a19c5174dfcf39798b42728ad56650497286379dc0c2c3/coverage-7.6.2-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:4eea60c79d36a8f39475b1af887663bc3ae4f31289cd216f514ce18d5938df40", size = 206888 },
|
222 |
+
{ url = "https://files.pythonhosted.org/packages/e0/de/820ecb42e892049c5f384430e98b35b899da3451dd0cdb2f867baf26abfa/coverage-7.6.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:aa68a6cdbe1bc6793a9dbfc38302c11599bbe1837392ae9b1d238b9ef3dafcf1", size = 207142 },
|
223 |
+
{ url = "https://files.pythonhosted.org/packages/dd/59/81fc7ad855d65eeb68fe9e7809cbb339946adb07be7ac32d3fc24dc17bd7/coverage-7.6.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3ec528ae69f0a139690fad6deac8a7d33629fa61ccce693fdd07ddf7e9931fba", size = 239658 },
|
224 |
+
{ url = "https://files.pythonhosted.org/packages/cd/a7/865de3eb9e78ffbf7afd92f86d2580b18edfb6f0481bd3c39b205e05a762/coverage-7.6.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ed5ac02126f74d190fa2cc14a9eb2a5d9837d5863920fa472b02eb1595cdc925", size = 236802 },
|
225 |
+
{ url = "https://files.pythonhosted.org/packages/36/94/3b8f3abf88b7c451f97fd14c98f536bcee364e74250d928d57cc97c38ddd/coverage-7.6.2-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:21c0ea0d4db8a36b275cb6fb2437a3715697a4ba3cb7b918d3525cc75f726304", size = 238793 },
|
226 |
+
{ url = "https://files.pythonhosted.org/packages/d5/4b/57f95e41a10525002f524f3dbd577a3a9871d67998f8a8eb192fe697dc7b/coverage-7.6.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:35a51598f29b2a19e26d0908bd196f771a9b1c5d9a07bf20be0adf28f1ad4f77", size = 238455 },
|
227 |
+
{ url = "https://files.pythonhosted.org/packages/99/c9/9fbe5b841628e1d9030c8044844afef4f4735586289eb9237eeb5b97f0d7/coverage-7.6.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:c9192925acc33e146864b8cf037e2ed32a91fdf7644ae875f5d46cd2ef086a5f", size = 236538 },
|
228 |
+
{ url = "https://files.pythonhosted.org/packages/43/0d/2200a0d447e30de94d48e4851c04d8dce37340815e7eda27457a7043c037/coverage-7.6.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:bf4eeecc9e10f5403ec06138978235af79c9a79af494eb6b1d60a50b49ed2869", size = 238383 },
|
229 |
+
{ url = "https://files.pythonhosted.org/packages/ec/8a/106c66faafb4a87002b698769d6de3c4db0b6c29a7aeb72de13b893c333e/coverage-7.6.2-cp313-cp313-win32.whl", hash = "sha256:e4ee15b267d2dad3e8759ca441ad450c334f3733304c55210c2a44516e8d5530", size = 209551 },
|
230 |
+
{ url = "https://files.pythonhosted.org/packages/c4/f5/1b39e2faaf5b9cc7eed568c444df5991ce7ff7138e2e735a6801be1bdadb/coverage-7.6.2-cp313-cp313-win_amd64.whl", hash = "sha256:c71965d1ced48bf97aab79fad56df82c566b4c498ffc09c2094605727c4b7e36", size = 210282 },
|
231 |
+
{ url = "https://files.pythonhosted.org/packages/79/a3/8dd4e6c09f5286094cd6c7edb115b3fbf06ad8304d45431722a4e3bc2508/coverage-7.6.2-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:7571e8bbecc6ac066256f9de40365ff833553e2e0c0c004f4482facb131820ef", size = 207629 },
|
232 |
+
{ url = "https://files.pythonhosted.org/packages/8e/db/a9aa7009bbdc570a235e1ac781c0a83aa323cac6db8f8f13c2127b110978/coverage-7.6.2-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:078a87519057dacb5d77e333f740708ec2a8f768655f1db07f8dfd28d7a005f0", size = 207902 },
|
233 |
+
{ url = "https://files.pythonhosted.org/packages/54/08/d0962be62d4335599ca2ff3a48bb68c9bfb80df74e28ca689ff5f392087b/coverage-7.6.2-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1e5e92e3e84a8718d2de36cd8387459cba9a4508337b8c5f450ce42b87a9e760", size = 250617 },
|
234 |
+
{ url = "https://files.pythonhosted.org/packages/a5/a2/158570aff1dd88b661a6c11281cbb190e8696e77798b4b2e47c74bfb2f39/coverage-7.6.2-cp313-cp313t-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ebabdf1c76593a09ee18c1a06cd3022919861365219ea3aca0247ededf6facd6", size = 246334 },
|
235 |
+
{ url = "https://files.pythonhosted.org/packages/aa/fe/b00428cca325b6585ca77422e4f64d7d86a225b14664b98682ea501efb57/coverage-7.6.2-cp313-cp313t-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:12179eb0575b8900912711688e45474f04ab3934aaa7b624dea7b3c511ecc90f", size = 248692 },
|
236 |
+
{ url = "https://files.pythonhosted.org/packages/30/21/0a15fefc13039450bc45e7159f3add92489f004555eb7dab9c7ad4365dd0/coverage-7.6.2-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:39d3b964abfe1519b9d313ab28abf1d02faea26cd14b27f5283849bf59479ff5", size = 248188 },
|
237 |
+
{ url = "https://files.pythonhosted.org/packages/de/b8/5c093526046a8450a7a3d62ad09517cf38e638f6b3ee9433dd6a73360501/coverage-7.6.2-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:84c4315577f7cd511d6250ffd0f695c825efe729f4205c0340f7004eda51191f", size = 246072 },
|
238 |
+
{ url = "https://files.pythonhosted.org/packages/1e/8b/542b607d2cff56e5a90a6948f5a9040b693761d2be2d3c3bf88957b02361/coverage-7.6.2-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:ff797320dcbff57caa6b2301c3913784a010e13b1f6cf4ab3f563f3c5e7919db", size = 247354 },
|
239 |
+
{ url = "https://files.pythonhosted.org/packages/95/82/2e9111aa5e59f42b332d387f64e3205c2263518d1e660154d0c9fc54390e/coverage-7.6.2-cp313-cp313t-win32.whl", hash = "sha256:2b636a301e53964550e2f3094484fa5a96e699db318d65398cfba438c5c92171", size = 210194 },
|
240 |
+
{ url = "https://files.pythonhosted.org/packages/9d/46/aabe4305cfc57cab4865f788ceceef746c422469720c32ed7a5b44e20f5e/coverage-7.6.2-cp313-cp313t-win_amd64.whl", hash = "sha256:d03a060ac1a08e10589c27d509bbdb35b65f2d7f3f8d81cf2fa199877c7bc58a", size = 211346 },
|
241 |
]
|
242 |
|
243 |
[package.optional-dependencies]
|
|
|
271 |
{ url = "https://files.pythonhosted.org/packages/d5/50/83c593b07763e1161326b3b8c6686f0f4b0f24d5526546bee538c89837d6/decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186", size = 9073 },
|
272 |
]
|
273 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
274 |
[[package]]
|
275 |
name = "diskcache"
|
276 |
version = "5.6.3"
|
|
|
291 |
|
292 |
[[package]]
|
293 |
name = "fastcore"
|
294 |
+
version = "1.7.12"
|
295 |
source = { registry = "https://pypi.org/simple" }
|
296 |
dependencies = [
|
297 |
{ name = "packaging" },
|
298 |
]
|
299 |
+
sdist = { url = "https://files.pythonhosted.org/packages/99/a0/a9805650fd1a8be2fa6bc53d00e77936e19402c95080a8e970045e029b23/fastcore-1.7.12.tar.gz", hash = "sha256:c0018b0be97ffc89408cf5b9239cd5b0c6aa8093e6d1495c77ea269128d0e0ec", size = 76820 }
|
300 |
wheels = [
|
301 |
+
{ url = "https://files.pythonhosted.org/packages/e4/c6/6f1532d45d1a3b9042fb79da80bfc7877210190825074a1244d288555fc9/fastcore-1.7.12-py3-none-any.whl", hash = "sha256:7436afea7c7ff247a689ba3a52c51be41d8d37249e47020ea26b2daff4b8d0c6", size = 80241 },
|
302 |
]
|
303 |
|
304 |
[[package]]
|
|
|
847 |
{ url = "https://files.pythonhosted.org/packages/13/a3/a812df4e2dd5696d1f351d58b8fe16a405b234ad2886a0dab9183fb78109/pycparser-2.22-py3-none-any.whl", hash = "sha256:c3702b6d3dd8c7abc1afa565d7e63d53a1d0bd86cdc24edd75470f4de499cfcc", size = 117552 },
|
848 |
]
|
849 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
850 |
[[package]]
|
851 |
name = "pyee"
|
852 |
version = "12.0.0"
|
|
|
868 |
{ url = "https://files.pythonhosted.org/packages/f7/3f/01c8b82017c199075f8f788d0d906b9ffbbc5a47dc9918a945e13d5a2bda/pygments-2.18.0-py3-none-any.whl", hash = "sha256:b8e6aca0523f3ab76fee51799c488e38782ac06eafcf95e7ba832985c8e7b13a", size = 1205513 },
|
869 |
]
|
870 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
871 |
[[package]]
|
872 |
name = "pytest"
|
873 |
version = "8.3.3"
|
|
|
947 |
|
948 |
[[package]]
|
949 |
name = "python-fasthtml"
|
950 |
+
version = "0.6.10"
|
951 |
source = { registry = "https://pypi.org/simple" }
|
952 |
dependencies = [
|
953 |
{ name = "beautifulsoup4" },
|
|
|
961 |
{ name = "starlette" },
|
962 |
{ name = "uvicorn", extra = ["standard"] },
|
963 |
]
|
964 |
+
sdist = { url = "https://files.pythonhosted.org/packages/f2/c1/7947e6e11c67b0836b1263f69cc70b519ca9b693df48174a0230d0330ed6/python-fasthtml-0.6.10.tar.gz", hash = "sha256:707f129bd73d2bc821fd90b2b90f6bb8874f0a8d2f2de28c4c3080b7d576c62c", size = 51655 }
|
965 |
wheels = [
|
966 |
+
{ url = "https://files.pythonhosted.org/packages/61/4c/df3e68018406fc0de8e05ec2f3691ef8c19050de0ed5bf337b6036c17b1e/python_fasthtml-0.6.10-py3-none-any.whl", hash = "sha256:54a685b90ced7a1d6ca275912a0a6fb00dd723a249e38ea81ea25fcfc421b8b1", size = 55635 },
|
967 |
]
|
968 |
|
969 |
[[package]]
|
|
|
1286 |
|
1287 |
[package.dev-dependencies]
|
1288 |
dev = [
|
|
|
1289 |
{ name = "ipykernel" },
|
1290 |
{ name = "lxml" },
|
1291 |
{ name = "pandas" },
|
|
|
1303 |
|
1304 |
[package.metadata.requires-dev]
|
1305 |
dev = [
|
|
|
1306 |
{ name = "ipykernel", specifier = ">=6.29.5" },
|
1307 |
{ name = "lxml", specifier = ">=5.3.0" },
|
1308 |
{ name = "pandas", specifier = ">=2.2.2" },
|