mirror of
https://github.com/ublue-os/forge.git
synced 2025-07-03 08:21:19 +03:00
wip - nicegui load config
This commit is contained in:
parent
85b0b3fdc5
commit
75095a841f
|
@ -1,59 +0,0 @@
|
||||||
#!/usr/bin/env python3
|
|
||||||
import asyncio
|
|
||||||
import os.path
|
|
||||||
import platform
|
|
||||||
import shlex
|
|
||||||
import sys
|
|
||||||
|
|
||||||
from nicegui import ui
|
|
||||||
|
|
||||||
async def run_command(command: str) -> None:
|
|
||||||
"""Run a command in the background and display the output in the pre-created dialog."""
|
|
||||||
dialog.open()
|
|
||||||
result.content = ''
|
|
||||||
command = command.replace('python3', sys.executable) # NOTE replace with machine-independent Python path (#1240)
|
|
||||||
process = await asyncio.create_subprocess_exec(
|
|
||||||
*shlex.split(command, posix='win' not in sys.platform.lower()),
|
|
||||||
stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.STDOUT,
|
|
||||||
cwd=os.path.dirname(os.path.abspath(__file__))
|
|
||||||
)
|
|
||||||
# NOTE we need to read the output in chunks, otherwise the process will block
|
|
||||||
output = ''
|
|
||||||
while True:
|
|
||||||
new = await process.stdout.read(4096)
|
|
||||||
if not new:
|
|
||||||
break
|
|
||||||
output += new.decode()
|
|
||||||
# NOTE the content of the markdown element is replaced every time we have new output
|
|
||||||
result.content = f'```\n{output}\n```'
|
|
||||||
|
|
||||||
with ui.header().classes(replace='row items-center') as header:
|
|
||||||
ui.button(on_click=lambda: left_drawer.toggle(), icon='menu').props('flat color=white')
|
|
||||||
with ui.tabs() as tabs:
|
|
||||||
ui.tab('Home')
|
|
||||||
ui.tab('Ansible')
|
|
||||||
ui.tab('Registry')
|
|
||||||
|
|
||||||
with ui.footer(value=False) as footer:
|
|
||||||
ui.label('Footer')
|
|
||||||
|
|
||||||
with ui.left_drawer().classes('bg-blue-100') as left_drawer:
|
|
||||||
ui.label('Side menu')
|
|
||||||
|
|
||||||
ui.button('Build Project', on_click=lambda: run_command('python3 runner.py')).props('no-caps')
|
|
||||||
with ui.dialog() as dialog, ui.card():
|
|
||||||
result = ui.markdown()
|
|
||||||
|
|
||||||
|
|
||||||
with ui.page_sticky(position='bottom-right', x_offset=20, y_offset=20):
|
|
||||||
ui.button(on_click=footer.toggle, icon='contact_support').props('fab')
|
|
||||||
|
|
||||||
with ui.tab_panels(tabs, value='A').classes('w-full'):
|
|
||||||
with ui.tab_panel('A'):
|
|
||||||
ui.label('Content of A')
|
|
||||||
with ui.tab_panel('B'):
|
|
||||||
ui.label('Content of B')
|
|
||||||
with ui.tab_panel('C'):
|
|
||||||
ui.label('Content of C')
|
|
||||||
|
|
||||||
ui.run()
|
|
|
@ -1,14 +0,0 @@
|
||||||
import sys
|
|
||||||
import ansible_runner
|
|
||||||
|
|
||||||
# run ansible/generic commands in interactive mode locally
|
|
||||||
out, err, rc = ansible_runner.run_command(
|
|
||||||
executable_cmd='ansible-playbook',
|
|
||||||
cmdline_args=['../playbooks/project_clone.yml', '-i', '../inventory.yml'],
|
|
||||||
input_fd=sys.stdin,
|
|
||||||
output_fd=sys.stdout,
|
|
||||||
error_fd=sys.stderr,
|
|
||||||
)
|
|
||||||
print("rc: {}".format(rc))
|
|
||||||
print("out: {}".format(out))
|
|
||||||
print("err: {}".format(err))
|
|
|
@ -1,88 +0,0 @@
|
||||||
# from nicegui import ui
|
|
||||||
|
|
||||||
# @ui.page('/')
|
|
||||||
# def page_layout():
|
|
||||||
# ui.label('CONTENT')
|
|
||||||
# [ui.label(f'Line {i}') for i in range(100)]
|
|
||||||
# with ui.header(elevated=True).style('background-color: #3874c8').classes('items-center justify-between'):
|
|
||||||
# ui.button(on_click=lambda: left_drawer.toggle(), icon='menu').props('flat color=white')
|
|
||||||
# ui.label('HEADER')
|
|
||||||
# with ui.left_drawer(fixed=False).style('background-color: #ebf1fa').props('bordered') as left_drawer:
|
|
||||||
# ui.label('LEFT DRAWER')
|
|
||||||
# with ui.footer().style('background-color: #3874c8'):
|
|
||||||
# ui.label('FOOTER')
|
|
||||||
|
|
||||||
# # ui.link('show page with fancy layout', page_layout)
|
|
||||||
|
|
||||||
# ui.run()
|
|
||||||
|
|
||||||
# from nicegui import ui
|
|
||||||
|
|
||||||
# with ui.header().classes(replace='row items-center') as header:
|
|
||||||
# ui.button(on_click=lambda: left_drawer.toggle(), icon='menu').props('flat color=white')
|
|
||||||
# with ui.tabs() as tabs:
|
|
||||||
# ui.tab('Home')
|
|
||||||
# ui.tab('Ansible')
|
|
||||||
# ui.tab('Registry')
|
|
||||||
|
|
||||||
# with ui.footer(value=False) as footer:
|
|
||||||
# ui.label('Footer')
|
|
||||||
|
|
||||||
# with ui.left_drawer().classes('bg-blue-100') as left_drawer:
|
|
||||||
# ui.label('Side menu')
|
|
||||||
|
|
||||||
# with ui.page_sticky(position='bottom-right', x_offset=20, y_offset=20):
|
|
||||||
# ui.button(on_click=footer.toggle, icon='contact_support').props('fab')
|
|
||||||
|
|
||||||
# with ui.tab_panels(tabs, value='A').classes('w-full'):
|
|
||||||
# with ui.tab_panel('A'):
|
|
||||||
# ui.label('Content of A')
|
|
||||||
# with ui.tab_panel('B'):
|
|
||||||
# ui.label('Content of B')
|
|
||||||
# with ui.tab_panel('C'):
|
|
||||||
# ui.label('Content of C')
|
|
||||||
|
|
||||||
# ui.run()
|
|
||||||
|
|
||||||
#!/usr/bin/env python3
|
|
||||||
import asyncio
|
|
||||||
import os.path
|
|
||||||
import platform
|
|
||||||
import shlex
|
|
||||||
import sys
|
|
||||||
|
|
||||||
from nicegui import ui
|
|
||||||
|
|
||||||
|
|
||||||
async def run_command(command: str) -> None:
|
|
||||||
"""Run a command in the background and display the output in the pre-created dialog."""
|
|
||||||
dialog.open()
|
|
||||||
result.content = ''
|
|
||||||
command = command.replace('python3', sys.executable) # NOTE replace with machine-independent Python path (#1240)
|
|
||||||
process = await asyncio.create_subprocess_exec(
|
|
||||||
*shlex.split(command, posix='win' not in sys.platform.lower()),
|
|
||||||
stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.STDOUT,
|
|
||||||
cwd=os.path.dirname(os.path.abspath(__file__))
|
|
||||||
)
|
|
||||||
# NOTE we need to read the output in chunks, otherwise the process will block
|
|
||||||
output = ''
|
|
||||||
while True:
|
|
||||||
new = await process.stdout.read(4096)
|
|
||||||
if not new:
|
|
||||||
break
|
|
||||||
output += new.decode()
|
|
||||||
# NOTE the content of the markdown element is replaced every time we have new output
|
|
||||||
result.content = f'```\n{output}\n```'
|
|
||||||
|
|
||||||
with ui.dialog() as dialog, ui.card():
|
|
||||||
result = ui.markdown()
|
|
||||||
|
|
||||||
ui.button('python3 runner.py', on_click=lambda: run_command('python3 runner.py')).props('no-caps')
|
|
||||||
ui.button('python3 slow.py', on_click=lambda: run_command('python3 slow.py')).props('no-caps')
|
|
||||||
with ui.row().classes('items-center'):
|
|
||||||
ui.button('python3 hello.py "<message>"', on_click=lambda: run_command(f'python3 hello.py "{message.value}"')) \
|
|
||||||
.props('no-caps')
|
|
||||||
message = ui.input('message', value='NiceGUI')
|
|
||||||
|
|
||||||
# NOTE: On Windows reload must be disabled to make asyncio.create_subprocess_exec work (see https://github.com/zauberzeug/nicegui/issues/486)
|
|
||||||
ui.run(reload=platform.system() != 'Windows')
|
|
0
anvil/nicegui/main.py
Executable file → Normal file
0
anvil/nicegui/main.py
Executable file → Normal file
|
@ -1,28 +1,65 @@
|
||||||
import ansible_runner
|
import ansible_runner
|
||||||
import re
|
import re
|
||||||
from nicegui import ui
|
from nicegui import ui
|
||||||
from utils import get_project_root
|
from utils import get_project_root, local_file_picker
|
||||||
|
from multiprocessing import Manager, Queue
|
||||||
|
|
||||||
|
ANSIBLE_EXTRA_VARS = None
|
||||||
|
|
||||||
|
|
||||||
# Ansible integration
|
# Ansible integration
|
||||||
def run_ansible_playbook(playbook_name: str, ngui_log: ui.log):
|
@ui.refreshable #https://www.reddit.com/r/nicegui/comments/1bphjk5/comment/kx7l5kj/?utm_source=share&utm_medium=web3x&utm_name=web3xcss&utm_term=1&utm_content=share_button
|
||||||
|
async def load_configuration_file() -> None:
|
||||||
|
result = await local_file_picker(
|
||||||
|
directory="/data", multiple=False, file_name_filter=".yml"
|
||||||
|
)
|
||||||
|
file_path = result[0]
|
||||||
|
with open(file_path, "r") as file:
|
||||||
|
data = file.read()
|
||||||
|
## Give feedback to user
|
||||||
|
ui.notify(f"You chose {file_path}")
|
||||||
|
## Display content
|
||||||
|
ui.code(content=data, language="yaml")
|
||||||
|
## Preserve configuration file path for ansible-playbook --extra-vars
|
||||||
|
ANSIBLE_EXTRA_VARS = f'"@{file_path}"'
|
||||||
|
|
||||||
|
async def run_ansible_playbook(playbook_name: str, ngui_log: ui.log):
|
||||||
project_root = str(get_project_root())
|
project_root = str(get_project_root())
|
||||||
playbook_path = project_root + "/ansible/playbooks/"
|
playbook_path = project_root + "/ansible/playbooks/"
|
||||||
inventory_path = project_root + "/ansible/inventory.yml"
|
inventory_path = project_root + "/ansible/inventory.yml"
|
||||||
response, error, return_code = ansible_runner.interface.run_command(
|
extra_vars_file = str(ANSIBLE_EXTRA_VARS)
|
||||||
|
thread, runner = ansible_runner.interface.run_command_async(
|
||||||
executable_cmd="ansible-playbook",
|
executable_cmd="ansible-playbook",
|
||||||
cmdline_args=[playbook_path + playbook_name, "-i", inventory_path],
|
cmdline_args=[
|
||||||
|
playbook_path + playbook_name,
|
||||||
|
"-i",
|
||||||
|
inventory_path,
|
||||||
|
"--extra-vars",
|
||||||
|
extra_vars_file,
|
||||||
|
]
|
||||||
)
|
)
|
||||||
# remove color characters from response until clear how to display them in a log
|
# clear log
|
||||||
ansi_escape = re.compile(r"\x1B(?:[@-Z\\-_]|\[[0-9;#]+[mGK]?)")
|
|
||||||
ansible_log = format(ansi_escape.sub("", response))
|
|
||||||
ngui_log.clear()
|
ngui_log.clear()
|
||||||
|
# regex to remove color characters from response until clear how to display them in a log
|
||||||
|
ansi_escape = re.compile(r"\x1B(?:[@-Z\\-_]|\[[0-9;#]+[mGK]?)")
|
||||||
|
# show log from asynchronous job
|
||||||
|
while runner.rc is None:
|
||||||
|
for event in runner.events:
|
||||||
|
ansible_log = format(ansi_escape.sub("", event['stdout']))
|
||||||
ngui_log.push(ansible_log)
|
ngui_log.push(ansible_log)
|
||||||
|
|
||||||
|
|
||||||
# Page content
|
# Page content
|
||||||
def content() -> None:
|
def content() -> None:
|
||||||
with ui.row().classes("w-full"):
|
with ui.row().classes("w-full"):
|
||||||
|
with ui.row().classes("w-full"):
|
||||||
|
with ui.card().classes("h-full"):
|
||||||
|
ui.button(text=
|
||||||
|
"Load configuration file",
|
||||||
|
on_click=lambda: load_configuration_file.refresh(),
|
||||||
|
icon="folder",
|
||||||
|
)
|
||||||
|
## show configuration file with ui.code
|
||||||
|
load_configuration_file()
|
||||||
with ui.row().classes("w-full"):
|
with ui.row().classes("w-full"):
|
||||||
# First Row
|
# First Row
|
||||||
with ui.card().classes("h-full"):
|
with ui.card().classes("h-full"):
|
||||||
|
@ -30,13 +67,13 @@ def content() -> None:
|
||||||
ui.button(
|
ui.button(
|
||||||
text="Clone project",
|
text="Clone project",
|
||||||
on_click=lambda: run_ansible_playbook(
|
on_click=lambda: run_ansible_playbook(
|
||||||
"project_clone.yml", ngui_log=log
|
playbook_name="project_clone.yml", ngui_log=playbook_log
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
ui.button(
|
ui.button(
|
||||||
text="Build project",
|
text="Build project",
|
||||||
on_click=lambda: run_ansible_playbook(
|
on_click=lambda: run_ansible_playbook(
|
||||||
"project_build.yml", ngui_log=log
|
"project_build.yml", ngui_log=playbook_log
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
# Second Row
|
# Second Row
|
||||||
|
@ -50,5 +87,5 @@ def content() -> None:
|
||||||
with ui.row().classes("w-full"):
|
with ui.row().classes("w-full"):
|
||||||
with ui.card().classes("w-full"):
|
with ui.card().classes("w-full"):
|
||||||
ui.label("Playbook Log").classes("text-h6")
|
ui.label("Playbook Log").classes("text-h6")
|
||||||
ui.button("Clear Log", on_click=lambda: log.clear())
|
ui.button("Clear Log", on_click=lambda: playbook_log.clear())
|
||||||
log = ui.log().classes("w-full h-full")
|
playbook_log = ui.log().classes("w-full h-full")
|
||||||
|
|
|
@ -1,4 +1,117 @@
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
import platform
|
||||||
|
from typing import Optional
|
||||||
|
from nicegui import events, ui
|
||||||
|
|
||||||
|
|
||||||
|
# most of the local_file_picker class is from https://github.com/zauberzeug/nicegui/blob/main/examples/local_file_picker/local_file_picker.py
|
||||||
|
# i just added the file_name_filter
|
||||||
|
class local_file_picker(ui.dialog):
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
directory: str,
|
||||||
|
*,
|
||||||
|
upper_limit: Optional[str] = ...,
|
||||||
|
multiple: bool = False,
|
||||||
|
show_hidden_files: bool = False,
|
||||||
|
file_name_filter: Optional[str] = None,
|
||||||
|
) -> None:
|
||||||
|
"""Local File Picker
|
||||||
|
|
||||||
|
This is a simple file picker that allows you to select a file from the local filesystem where NiceGUI is running.
|
||||||
|
|
||||||
|
:param directory: The directory to start in.
|
||||||
|
:param upper_limit: The directory to stop at (None: no limit, default: same as the starting directory).
|
||||||
|
:param multiple: Whether to allow multiple files to be selected.
|
||||||
|
:param show_hidden_files: Whether to show hidden files.
|
||||||
|
:param file_name_filter: Filter files in directory based on file name
|
||||||
|
"""
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
|
self.path = Path(directory).expanduser()
|
||||||
|
if upper_limit is None:
|
||||||
|
self.upper_limit = None
|
||||||
|
else:
|
||||||
|
self.upper_limit = Path(
|
||||||
|
directory if upper_limit == ... else upper_limit
|
||||||
|
).expanduser()
|
||||||
|
self.show_hidden_files = show_hidden_files
|
||||||
|
self.file_name_filter = file_name_filter
|
||||||
|
with self, ui.card():
|
||||||
|
self.add_drives_toggle()
|
||||||
|
self.grid = (
|
||||||
|
ui.aggrid(
|
||||||
|
{
|
||||||
|
"columnDefs": [{"field": "name", "headerName": "File"}],
|
||||||
|
"rowSelection": "multiple" if multiple else "single",
|
||||||
|
},
|
||||||
|
html_columns=[0],
|
||||||
|
)
|
||||||
|
.classes("w-96")
|
||||||
|
.on("cellDoubleClicked", self.handle_double_click)
|
||||||
|
)
|
||||||
|
with ui.row().classes("w-full justify-end"):
|
||||||
|
ui.button("Cancel", on_click=self.close).props("outline")
|
||||||
|
ui.button("Ok", on_click=self._handle_ok)
|
||||||
|
self.update_grid()
|
||||||
|
|
||||||
|
def add_drives_toggle(self):
|
||||||
|
if platform.system() == "Windows":
|
||||||
|
import win32api
|
||||||
|
|
||||||
|
drives = win32api.GetLogicalDriveStrings().split("\000")[:-1]
|
||||||
|
self.drives_toggle = ui.toggle(
|
||||||
|
drives, value=drives[0], on_change=self.update_drive
|
||||||
|
)
|
||||||
|
|
||||||
|
def update_drive(self):
|
||||||
|
self.path = Path(self.drives_toggle.value).expanduser()
|
||||||
|
self.update_grid()
|
||||||
|
|
||||||
|
def update_grid(self) -> None:
|
||||||
|
paths = list(self.path.glob("*"))
|
||||||
|
if not self.show_hidden_files:
|
||||||
|
paths = [p for p in paths if not p.name.startswith(".")]
|
||||||
|
if self.file_name_filter: # Add this condition
|
||||||
|
paths = [p for p in paths if p.is_file() and p.suffix == self.file_name_filter]
|
||||||
|
paths.sort(key=lambda p: p.name.lower())
|
||||||
|
paths.sort(key=lambda p: not p.is_dir())
|
||||||
|
|
||||||
|
self.grid.options["rowData"] = [
|
||||||
|
{
|
||||||
|
"name": f"📁 <strong>{p.name}</strong>" if p.is_dir() else p.name,
|
||||||
|
"path": str(p),
|
||||||
|
}
|
||||||
|
for p in paths
|
||||||
|
]
|
||||||
|
if (
|
||||||
|
self.upper_limit is None
|
||||||
|
and self.path != self.path.parent
|
||||||
|
or self.upper_limit is not None
|
||||||
|
and self.path != self.upper_limit
|
||||||
|
):
|
||||||
|
self.grid.options["rowData"].insert(
|
||||||
|
0,
|
||||||
|
{
|
||||||
|
"name": "📁 <strong>..</strong>",
|
||||||
|
"path": str(self.path.parent),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
self.grid.update()
|
||||||
|
|
||||||
|
def handle_double_click(self, e: events.GenericEventArguments) -> None:
|
||||||
|
self.path = Path(e.args["data"]["path"])
|
||||||
|
if self.path.is_dir():
|
||||||
|
self.update_grid()
|
||||||
|
else:
|
||||||
|
self.submit([str(self.path)])
|
||||||
|
|
||||||
|
async def _handle_ok(self):
|
||||||
|
rows = await ui.run_javascript(
|
||||||
|
f"getElement({self.grid.id}).gridOptions.api.getSelectedRows()"
|
||||||
|
)
|
||||||
|
self.submit([r["path"] for r in rows])
|
||||||
|
|
||||||
|
|
||||||
def get_project_root() -> Path:
|
def get_project_root() -> Path:
|
||||||
|
|
|
@ -107,6 +107,9 @@ spec:
|
||||||
- name: ublue-os_forge-certs-pvc
|
- name: ublue-os_forge-certs-pvc
|
||||||
persistentVolumeClaim:
|
persistentVolumeClaim:
|
||||||
claimName: ublue-os_forge-certs
|
claimName: ublue-os_forge-certs
|
||||||
|
- name: ublue-os_forge-data-pvc
|
||||||
|
persistentVolumeClaim:
|
||||||
|
claimName: ublue-os_forge-data
|
||||||
containers:
|
containers:
|
||||||
- name: ansible.${FORGE_DOMAIN_NAME}
|
- name: ansible.${FORGE_DOMAIN_NAME}
|
||||||
image: anvil # will be built on pod start
|
image: anvil # will be built on pod start
|
||||||
|
@ -114,6 +117,8 @@ spec:
|
||||||
- mountPath: /certs
|
- mountPath: /certs
|
||||||
name: ublue-os_forge-certs-pvc
|
name: ublue-os_forge-certs-pvc
|
||||||
readOnly: true
|
readOnly: true
|
||||||
|
- mountPath: /data
|
||||||
|
name: ublue-os_forge-data-pvc
|
||||||
env:
|
env:
|
||||||
- name: ANSIBLE_HOST_USER
|
- name: ANSIBLE_HOST_USER
|
||||||
valueFrom:
|
valueFrom:
|
||||||
|
|
Loading…
Reference in a new issue