Skip to content
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
5aaa249
including notes
elnelson575 Aug 25, 2025
cc15245
Removing notes to self
elnelson575 Aug 25, 2025
e6b8260
cleanup
elnelson575 Aug 25, 2025
0fb5ccf
Merge branch 'main' into fix/js_css_inclusion
elnelson575 Aug 25, 2025
f4c53fd
With prints
elnelson575 Aug 26, 2025
2764cbe
Add multifile example, notes
elnelson575 Aug 26, 2025
03839f0
Fixing types
elnelson575 Aug 26, 2025
649c856
Merge branch 'fix/js_css_inclusion' of github.com:posit-dev/py-shiny …
elnelson575 Aug 26, 2025
c1a11a1
Updating tests
elnelson575 Aug 27, 2025
ff8581e
Debugging
elnelson575 Aug 27, 2025
c888903
fixes
elnelson575 Aug 27, 2025
e99f795
Deleting old test
elnelson575 Aug 27, 2025
c1d7568
Merging in changes from other branch
elnelson575 Aug 27, 2025
14012c8
Added comments, minor changes
elnelson575 Aug 28, 2025
7ffeab5
Relinking vs inline
elnelson575 Aug 28, 2025
63a3c40
Update shiny/ui/_include_helpers.py
elnelson575 Aug 28, 2025
6609d20
Updating prints
elnelson575 Aug 28, 2025
b477812
Updating 'shiny_include_files to have 777 perms
elnelson575 Aug 29, 2025
30fc4f7
Testing only new dir change
elnelson575 Sep 2, 2025
87afb30
Updating to deal with 404
elnelson575 Sep 3, 2025
8ae6460
Updating one missing piece
elnelson575 Sep 3, 2025
74c931e
Removing prints
elnelson575 Sep 4, 2025
26fedcc
Updatign changelog
elnelson575 Sep 4, 2025
b48aea9
Minor edit
elnelson575 Sep 4, 2025
88a9fee
Merge branch 'main' into fix/js_css_include_perms
elnelson575 Sep 4, 2025
e4fea42
Updating tests
elnelson575 Sep 4, 2025
cee6f3e
Updated changelog and comment placement
elnelson575 Sep 4, 2025
9004f58
Updating test name
elnelson575 Sep 4, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
29 changes: 25 additions & 4 deletions shiny/ui/_include_helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
# as the app, the app's source will be included. Should we just not copy .py/.r files?


# TODO: This does throw js errors if you're loading in other stuff depending on your config
@add_example()
def include_js(
path: Path | str,
Expand Down Expand Up @@ -214,37 +215,57 @@ def create_include_dependency(

def maybe_copy_files(path: Path | str, include_files: bool) -> tuple[str, str]:
hash = get_hash(path, include_files)
print("path: ", path, "Hash: ", hash)

# To avoid unnecessary work when the same file is included multiple times,
# use a directory scoped by a hash of the file.
tmpdir = os.path.join(tempfile.gettempdir(), "shiny_include_files", hash)
path_dest = os.path.join(tmpdir, os.path.basename(path))
print("path_dest:", path_dest)

# Since the hash/tmpdir should represent all the files in the path's directory,
# we can simply return here
# we can check if it exists to determine if we have a cache hit
if os.path.exists(path_dest):
print("Path already exists:", path_dest)
return path_dest, hash

# Otherwise, make sure we have a clean slate
if os.path.exists(tmpdir):
print("Folder already exists, but not files, removing.")
shutil.rmtree(tmpdir)

# This recursively changes permissions to 755 (owner rwx, other rx) for all files
# under `tmp/unique_hash` so that the app can be run by other collaborators
# on a multi-tenant system. See #2061
if include_files:
shutil.copytree(os.path.dirname(path), tmpdir)
for dirpath, dirs, filenames in os.walk(tmpdir):
# set perms on files
for file in filenames:
os.chmod(os.path.join(dirpath, file), 0o755)
print(
"File: ",
file,
"File perms: ",
oct(os.stat(os.path.join(dirpath, file)).st_mode),
)
else:
os.makedirs(tmpdir, exist_ok=True)
os.mkdir(tmpdir, mode=0o755)
shutil.copy(path, path_dest)
os.chmod(path_dest, 0o755)
print("File: ", path_dest, "perms", oct(os.stat(path_dest).st_mode))

return path_dest, hash


def get_hash(path: Path | str, include_files: bool) -> str:
if include_files:
key = get_file_key(path)
else:
dir = os.path.dirname(path)
files = glob.iglob(os.path.join(dir, "**"), recursive=True)
key = "\n".join([get_file_key(x) for x in files])
else:
key = get_file_key(path)

return hash_deterministic(key)


Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
from pathlib import Path

from shiny import App, Inputs, Outputs, Session, ui

js_file = Path(__file__).parent / "js" / "customjs.js"
css_file = Path(__file__).parent / "css" / "style.css"

# Define the UI
app_ui = ui.page_fluid(
ui.include_css(css_file, method="link_files"),
ui.include_js(js_file, method="link_files"),
ui.h1("Simple Shiny App with External CSS"),
ui.div(
ui.p("This is a simple Shiny app that demonstrates ui.include_css()"),
ui.p("The styling comes from an external CSS file!"),
class_="content",
),
)


# Define the server
def server(input: Inputs, output: Outputs, session: Session):
pass


# Create and run the app
app = App(app_ui, server)
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
body {
background-color: #c8e1f7;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
@import url("./evenmorecss/more.css");

body {
font-family: Arial, sans-serif;
}

h1 {
color: black;
border-bottom: 2px solid #4682b4;
padding-bottom: 10px;
}

.content {
margin: 20px;
padding: 15px;
background-color: white;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
const newParagraph = document.createElement('p');
newParagraph.textContent = 'Heyo!';
const bodyElement = document.body;
bodyElement.appendChild(newParagraph);
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
from playwright.sync_api import Page, expect

from shiny.run import ShinyAppProc


def test_inclusion(page: Page, local_app: ShinyAppProc) -> None:
page.goto(local_app.url)

expect(page.locator("body > p")).to_have_text("Heyo!")
Copy link
Collaborator

@cpsievert cpsievert Sep 4, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This test is for the other PR, right?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh nvm, I think I see why we're adding the test -- it's not that we're directly testing this fix, it's more about adding coverage that should've already existed?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, because we're altering how we include files, I want to have a test for both "link_files" and "link" versions

Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
from pathlib import Path

from shiny import App, Inputs, Outputs, Session, ui

js_file = Path(__file__).parent / "customjs.js"
css_file = Path(__file__).parent / "style.css"

# Define the UI
app_ui = ui.page_fluid(
ui.include_css(css_file, method="link"),
ui.include_js(js_file, method="link"),
ui.h1("Simple Shiny App with External CSS"),
ui.div(
ui.p("This is a simple Shiny app that demonstrates ui.include_css()"),
ui.p("The styling comes from an external CSS file!"),
class_="content",
),
)


# Define the server
def server(input: Inputs, output: Outputs, session: Session):
pass


# Create and run the app
app = App(app_ui, server)
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
const newParagraph = document.createElement('p');
newParagraph.textContent = 'Heyo!';
const bodyElement = document.body;
bodyElement.appendChild(newParagraph);
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
body {
background-color: #70bfef;
font-family: Arial, sans-serif;
}

h1 {
color: black;
border-bottom: 2px solid #4682b4;
padding-bottom: 10px;
}

.content {
margin: 20px;
padding: 15px;
background-color: white;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
from playwright.sync_api import Page, expect

from shiny.run import ShinyAppProc


def test_inclusion(page: Page, local_app: ShinyAppProc) -> None:
page.goto(local_app.url)

expect(page.locator("body > p")).to_have_text("Heyo!")
Loading