portacode 1.4.35.dev0__tar.gz → 1.4.35.dev2__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/PKG-INFO +1 -1
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/connect.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/_version.py +2 -2
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/cli.py +13 -3
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/automation_v2_handlers.py +107 -1
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/update_handler.py +6 -9
- portacode-1.4.35.dev2/portacode/test_updater.py +100 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/updater.py +48 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/PKG-INFO +1 -1
- portacode-1.4.35.dev0/portacode/test_updater.py +0 -44
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/.claude/agents/communication-manager.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/.claude/settings.local.json +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/.gitignore +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/.gitmodules +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/LICENSE +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/MANIFEST.in +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/Makefile +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/backup.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/build_android.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/connect.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docker-compose.yaml +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/cloudflared-domain-connect-containers-audit.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/creative-team-brief-portacode.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/devops-messaging-ab-tests.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/homepage-dashboard-positioning-fixes.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/images/device-transfer-button.png +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/images/device-transfer-modal.png +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/images/pair-device-button.png +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/images/pairing-request.png +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/images/student-workspace.png +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/template-guide.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/simple_device/Dockerfile +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/simple_device/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/simple_device/data/device-01/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/simple_device/data/device-01/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/simple_device/docker-compose.yaml +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/Dockerfile +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/.local/share/portacode/run/gateway.pid +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/.gitignore +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/db.sqlite3 +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/__pycache__/__init__.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/__pycache__/settings.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/__pycache__/urls.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/__pycache__/wsgi.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/__init__.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/admin.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/apps.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/menu.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/models.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/urls.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/__pycache__/views.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/migrations/__pycache__/__init__.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-01/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-02/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-03/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-04/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-05/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-06/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-07/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-08/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-09/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/.local/share/portacode/keys/id_portacode +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/.local/share/portacode/keys/id_portacode.pub +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/data/student-10/workspace/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/docker-compose.yaml +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/galactic_bakeshop/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/galactic_bakeshop/asgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/galactic_bakeshop/settings.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/galactic_bakeshop/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/galactic_bakeshop/wsgi.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/manage.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/templates/treats/home.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/admin.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/apps.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/menu.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/migrations/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/urls.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/initial_content/treats/views.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/examples/workshop_fleet/instructions/WELCOME.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/__main__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/client.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/WEBSOCKET_PROTOCOL.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/base.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/chunked_content.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/cloudflare_forwarding.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/cloudflare_tunnel.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/diff_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/file_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_aware_file_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/file_system_watcher.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/git_manager.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/manager.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/models.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state/utils.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/project_state_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/proxmox_infra.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/registry.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/session.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/system_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/tab_factory.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/terminal_handlers.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/test_proxmox_infra.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/multiplex.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/terminal.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/data.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/keypair.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/__pycache__/__init__.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/__pycache__/link_capture_wrapper.cpython-311.pyc +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/elinks +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/gio-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/gnome-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/gvfs-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/kde-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/kfmclient +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/link_capture_exec.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/link_capture_wrapper.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/links +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/links2 +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/lynx +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/mate-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/netsurf +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/sensible-browser +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/w3m +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/x-www-browser +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/link_capture/bin/xdg-open +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/logging_categories.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/pairing.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/restart.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/service.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/static/js/test-ntp-clock.html +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/static/js/utils/ntp-clock.js +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/cloudflared_login.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/ensure_cloudflared.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/ensure_pyyaml.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/forwarding_state.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/get_domain.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/privileged.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/service_install.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/tunneling/state.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/utils/NTP_ARCHITECTURE.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/utils/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/utils/diff_apply.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/utils/diff_renderer.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/utils/ntp_clock.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/SOURCES.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/dependency_links.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/entry_points.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/requires.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode.egg-info/top_level.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/pyproject.toml +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/restore.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/run_tests.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/setup.cfg +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/setup.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_device_online.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_file_operations.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_git_status_ui.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_login_flow.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_navigate_testing_folder.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_play_store_screenshots.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_terminal_buffer_performance.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_terminal_interaction.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_terminal_loading_race_condition.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_modules/test_terminal_start.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/test_request_id.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/.env.example +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/README.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/cli.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/__init__.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/base_test.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/cli_manager.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/hierarchical_runner.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/playwright_manager.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/runner.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/shared_cli_manager.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/core/test_discovery.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/testing_framework/requirements.txt +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/UI_UX/opening_a_file_on_desktop_results_in_nothing.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/UI_UX/server_occasionally_stops_communicating_with_all_devices.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/agent_context_management.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/django_server_time_sync.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/device_performance_degradation.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/git_data_not_captured_in_proxmox.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/indefinite_resource_loading.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/portacode_service_silently_down.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/premature_terminal_exit.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/project_cpu_hotspots.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/terminals_exit_upon_starting.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/issues/wrong_item_classification_on_client_side.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/todo/smartphone_terminal_input_frustrations.md +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/tools/generate_play_store_assets.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/tools/pairing_tester.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/tools/run_screenshot_suite.sh +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/tools/test_python_ntp_clock.py +0 -0
- {portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/validate.sh +0 -0
|
File without changes
|
|
@@ -28,7 +28,7 @@ version_tuple: VERSION_TUPLE
|
|
|
28
28
|
commit_id: COMMIT_ID
|
|
29
29
|
__commit_id__: COMMIT_ID
|
|
30
30
|
|
|
31
|
-
__version__ = version = '1.4.35.
|
|
32
|
-
__version_tuple__ = version_tuple = (1, 4, 35, '
|
|
31
|
+
__version__ = version = '1.4.35.dev2'
|
|
32
|
+
__version_tuple__ = version_tuple = (1, 4, 35, 'dev2')
|
|
33
33
|
|
|
34
34
|
__commit_id__ = commit_id = None
|
|
@@ -24,7 +24,7 @@ from .keypair import (
|
|
|
24
24
|
)
|
|
25
25
|
from .pairing import PairingError, pair_device_with_code
|
|
26
26
|
from .connection.client import ConnectionManager, run_until_interrupt
|
|
27
|
-
from .updater import build_pip_install_command
|
|
27
|
+
from .updater import build_pip_install_command, run_pip_install_command
|
|
28
28
|
|
|
29
29
|
GATEWAY_URL = "wss://portacode.com/gateway"
|
|
30
30
|
GATEWAY_ENV = "PORTACODE_GATEWAY"
|
|
@@ -638,8 +638,18 @@ def restart_command(method: str, verbose: bool) -> None:
|
|
|
638
638
|
def set_version_command(version: str) -> None:
|
|
639
639
|
"""Install a specific Portacode version and restart the system service."""
|
|
640
640
|
click.echo(f"Installing Portacode {version}…")
|
|
641
|
-
|
|
642
|
-
|
|
641
|
+
interactive_tty = sys.stdin.isatty() and sys.stdout.isatty()
|
|
642
|
+
if interactive_tty and os.geteuid() != 0:
|
|
643
|
+
click.echo(
|
|
644
|
+
click.style(
|
|
645
|
+
"[sudo] If required, Portacode will attempt passwordless sudo first and then prompt for your password.",
|
|
646
|
+
fg="yellow",
|
|
647
|
+
)
|
|
648
|
+
)
|
|
649
|
+
result = run_pip_install_command(
|
|
650
|
+
build_pip_install_command(version=version),
|
|
651
|
+
allow_sudo_fallback=True,
|
|
652
|
+
interactive_sudo=interactive_tty,
|
|
643
653
|
)
|
|
644
654
|
if result.returncode != 0:
|
|
645
655
|
error_msg = (result.stderr or result.stdout or "").strip() or "unknown error"
|
|
@@ -36,6 +36,7 @@ EXPOSED_SERVICES_ENV_KEY = "PORTACODE_EXPOSED_SERVICES_JSON"
|
|
|
36
36
|
WAIT_FOR_PLACEHOLDER_RE = re.compile(r"\[exposed:(\d{1,5})\]", flags=re.IGNORECASE)
|
|
37
37
|
WAIT_FOR_STEP_INTERVAL_SECONDS = 3.0
|
|
38
38
|
WAIT_FOR_REQUEST_TIMEOUT_SECONDS = 5.0
|
|
39
|
+
WAIT_FOR_DEFAULT_TIMEOUT_SECONDS = 120.0
|
|
39
40
|
|
|
40
41
|
|
|
41
42
|
def _trim_text(value: Any, max_chars: int = MAX_STDIO_CHARS) -> str:
|
|
@@ -321,10 +322,19 @@ class _AutomationRuntimeV2:
|
|
|
321
322
|
self._persist_state()
|
|
322
323
|
await self._notify_change(dict(state))
|
|
323
324
|
default_timeout = float(state.get("default_timeout_seconds") or DEFAULT_STEP_TIMEOUT_SECONDS)
|
|
324
|
-
|
|
325
|
+
timeout_fallback = WAIT_FOR_DEFAULT_TIMEOUT_SECONDS if wait_for_target else default_timeout
|
|
326
|
+
timeout_seconds = _extract_step_timeout(step, timeout_fallback)
|
|
327
|
+
task_id_payload: Any = int(task_id) if str(task_id).isdigit() else str(task_id)
|
|
325
328
|
|
|
326
329
|
if wait_for_target:
|
|
327
330
|
command_label = f"wait_for {wait_for_target}"
|
|
331
|
+
logger.info(
|
|
332
|
+
"automation_v2 wait_for start task=%s step=%s target=%s timeout_s=%.1f",
|
|
333
|
+
task_id,
|
|
334
|
+
step_index,
|
|
335
|
+
wait_for_target,
|
|
336
|
+
timeout_seconds,
|
|
337
|
+
)
|
|
328
338
|
duration_start = time.monotonic()
|
|
329
339
|
wait_result = await self._run_wait_for_step(
|
|
330
340
|
target=wait_for_target,
|
|
@@ -391,6 +401,14 @@ class _AutomationRuntimeV2:
|
|
|
391
401
|
return
|
|
392
402
|
|
|
393
403
|
if timed_out or returncode != 0:
|
|
404
|
+
logger.warning(
|
|
405
|
+
"automation_v2 wait_for failed task=%s step=%s target=%s resolved_url=%s stderr=%s",
|
|
406
|
+
task_id,
|
|
407
|
+
step_index,
|
|
408
|
+
wait_for_target,
|
|
409
|
+
resolved_url,
|
|
410
|
+
stderr_text,
|
|
411
|
+
)
|
|
394
412
|
state["status"] = "failed"
|
|
395
413
|
state["current_step_status"] = "failed"
|
|
396
414
|
state["completed_at"] = time.time()
|
|
@@ -406,6 +424,14 @@ class _AutomationRuntimeV2:
|
|
|
406
424
|
state["state_seq"] = int(state.get("state_seq") or 0) + 1
|
|
407
425
|
self._persist_state()
|
|
408
426
|
await self._notify_change(dict(state))
|
|
427
|
+
logger.info(
|
|
428
|
+
"automation_v2 wait_for success task=%s step=%s target=%s resolved_url=%s duration_s=%.3f",
|
|
429
|
+
task_id,
|
|
430
|
+
step_index,
|
|
431
|
+
wait_for_target,
|
|
432
|
+
resolved_url,
|
|
433
|
+
duration_s,
|
|
434
|
+
)
|
|
409
435
|
continue
|
|
410
436
|
|
|
411
437
|
start = time.monotonic()
|
|
@@ -648,6 +674,13 @@ class _AutomationRuntimeV2:
|
|
|
648
674
|
automation_task_id: Any,
|
|
649
675
|
automation_step_index: int,
|
|
650
676
|
) -> dict[str, Any]:
|
|
677
|
+
logger.info(
|
|
678
|
+
"automation_v2 wait_for polling begin task=%s step=%s target=%s timeout_s=%.1f",
|
|
679
|
+
automation_task_id,
|
|
680
|
+
automation_step_index,
|
|
681
|
+
target,
|
|
682
|
+
timeout_seconds,
|
|
683
|
+
)
|
|
651
684
|
deadline = time.monotonic() + max(1.0, float(timeout_seconds))
|
|
652
685
|
attempts = 0
|
|
653
686
|
last_error = ""
|
|
@@ -658,9 +691,39 @@ class _AutomationRuntimeV2:
|
|
|
658
691
|
resolved_url = self._resolve_wait_for_url(target)
|
|
659
692
|
except Exception as exc:
|
|
660
693
|
last_error = str(exc)
|
|
694
|
+
logger.warning(
|
|
695
|
+
"automation_v2 wait_for resolve failed task=%s step=%s attempt=%s target=%s error=%s",
|
|
696
|
+
automation_task_id,
|
|
697
|
+
automation_step_index,
|
|
698
|
+
attempts,
|
|
699
|
+
target,
|
|
700
|
+
last_error,
|
|
701
|
+
)
|
|
702
|
+
if self._event_sender is not None:
|
|
703
|
+
try:
|
|
704
|
+
await self._event_sender(
|
|
705
|
+
{
|
|
706
|
+
"event": "terminal_exec_output",
|
|
707
|
+
"command": f"wait_for {target}",
|
|
708
|
+
"stderr": (
|
|
709
|
+
f"wait_for attempt={attempts} resolve_failed error={last_error}\n"
|
|
710
|
+
),
|
|
711
|
+
"automation_task_id": automation_task_id,
|
|
712
|
+
"automation_step_index": automation_step_index,
|
|
713
|
+
}
|
|
714
|
+
)
|
|
715
|
+
except Exception:
|
|
716
|
+
logger.exception("automation_v2: failed to emit terminal_exec_output")
|
|
661
717
|
await asyncio.sleep(WAIT_FOR_STEP_INTERVAL_SECONDS)
|
|
662
718
|
continue
|
|
663
719
|
|
|
720
|
+
logger.info(
|
|
721
|
+
"automation_v2 wait_for probe task=%s step=%s attempt=%s url=%s",
|
|
722
|
+
automation_task_id,
|
|
723
|
+
automation_step_index,
|
|
724
|
+
attempts,
|
|
725
|
+
resolved_url,
|
|
726
|
+
)
|
|
664
727
|
ok, status, error = await asyncio.to_thread(self._probe_http_url, resolved_url)
|
|
665
728
|
if ok:
|
|
666
729
|
message = (
|
|
@@ -668,6 +731,14 @@ class _AutomationRuntimeV2:
|
|
|
668
731
|
if status is not None
|
|
669
732
|
else f"wait_for success url={resolved_url} attempts={attempts}"
|
|
670
733
|
)
|
|
734
|
+
logger.info(
|
|
735
|
+
"automation_v2 wait_for probe succeeded task=%s step=%s attempt=%s url=%s status=%s",
|
|
736
|
+
automation_task_id,
|
|
737
|
+
automation_step_index,
|
|
738
|
+
attempts,
|
|
739
|
+
resolved_url,
|
|
740
|
+
status,
|
|
741
|
+
)
|
|
671
742
|
if self._event_sender is not None:
|
|
672
743
|
try:
|
|
673
744
|
await self._event_sender(
|
|
@@ -689,8 +760,43 @@ class _AutomationRuntimeV2:
|
|
|
689
760
|
"resolved_url": resolved_url,
|
|
690
761
|
}
|
|
691
762
|
last_error = error or (f"http_status={status}" if status is not None else "request failed")
|
|
763
|
+
logger.info(
|
|
764
|
+
"automation_v2 wait_for probe not-ready task=%s step=%s attempt=%s url=%s status=%s error=%s",
|
|
765
|
+
automation_task_id,
|
|
766
|
+
automation_step_index,
|
|
767
|
+
attempts,
|
|
768
|
+
resolved_url,
|
|
769
|
+
status,
|
|
770
|
+
last_error,
|
|
771
|
+
)
|
|
772
|
+
if self._event_sender is not None:
|
|
773
|
+
try:
|
|
774
|
+
await self._event_sender(
|
|
775
|
+
{
|
|
776
|
+
"event": "terminal_exec_output",
|
|
777
|
+
"command": f"wait_for {target}",
|
|
778
|
+
"stderr": (
|
|
779
|
+
f"wait_for attempt={attempts} url={resolved_url} "
|
|
780
|
+
f"status={status if status is not None else 'n/a'} "
|
|
781
|
+
f"error={last_error}\n"
|
|
782
|
+
),
|
|
783
|
+
"automation_task_id": automation_task_id,
|
|
784
|
+
"automation_step_index": automation_step_index,
|
|
785
|
+
}
|
|
786
|
+
)
|
|
787
|
+
except Exception:
|
|
788
|
+
logger.exception("automation_v2: failed to emit terminal_exec_output")
|
|
692
789
|
await asyncio.sleep(WAIT_FOR_STEP_INTERVAL_SECONDS)
|
|
693
790
|
|
|
791
|
+
logger.warning(
|
|
792
|
+
"automation_v2 wait_for timeout task=%s step=%s target=%s resolved_url=%s attempts=%s last_error=%s",
|
|
793
|
+
automation_task_id,
|
|
794
|
+
automation_step_index,
|
|
795
|
+
target,
|
|
796
|
+
resolved_url,
|
|
797
|
+
attempts,
|
|
798
|
+
last_error,
|
|
799
|
+
)
|
|
694
800
|
return {
|
|
695
801
|
"returncode": 1,
|
|
696
802
|
"stdout": "",
|
{portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/portacode/connection/handlers/update_handler.py
RENAMED
|
@@ -1,10 +1,9 @@
|
|
|
1
1
|
"""Update handler for Portacode CLI."""
|
|
2
2
|
|
|
3
3
|
import logging
|
|
4
|
-
import subprocess
|
|
5
4
|
from typing import Any, Dict, Optional
|
|
6
5
|
|
|
7
|
-
from portacode.updater import build_pip_install_command
|
|
6
|
+
from portacode.updater import build_pip_install_command, run_pip_install_command
|
|
8
7
|
from .base import AsyncHandler
|
|
9
8
|
|
|
10
9
|
logger = logging.getLogger(__name__)
|
|
@@ -22,7 +21,11 @@ class UpdatePortacodeHandler(AsyncHandler):
|
|
|
22
21
|
try:
|
|
23
22
|
logger.info("Starting Portacode CLI update...")
|
|
24
23
|
pip_cmd = build_pip_install_command()
|
|
25
|
-
result =
|
|
24
|
+
result = run_pip_install_command(
|
|
25
|
+
pip_cmd,
|
|
26
|
+
allow_sudo_fallback=True,
|
|
27
|
+
interactive_sudo=False,
|
|
28
|
+
)
|
|
26
29
|
|
|
27
30
|
if result.returncode != 0:
|
|
28
31
|
error_msg = result.stderr.strip() or result.stdout.strip()
|
|
@@ -50,12 +53,6 @@ class UpdatePortacodeHandler(AsyncHandler):
|
|
|
50
53
|
request_restart(method="auto", in_service=True)
|
|
51
54
|
return None
|
|
52
55
|
|
|
53
|
-
except subprocess.TimeoutExpired:
|
|
54
|
-
return {
|
|
55
|
-
"event": "update_portacode_response",
|
|
56
|
-
"success": False,
|
|
57
|
-
"error": "Update timed out after 120 seconds",
|
|
58
|
-
}
|
|
59
56
|
except Exception as e:
|
|
60
57
|
logger.exception("Update failed with exception")
|
|
61
58
|
return {
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
from unittest import TestCase
|
|
2
|
+
from unittest.mock import patch
|
|
3
|
+
import subprocess
|
|
4
|
+
|
|
5
|
+
from portacode.updater import build_pip_install_command, run_pip_install_command
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
class UpdaterCommandTests(TestCase):
|
|
9
|
+
def _cp(self, returncode: int, stdout: str = "", stderr: str = "") -> subprocess.CompletedProcess[str]:
|
|
10
|
+
return subprocess.CompletedProcess(args=["cmd"], returncode=returncode, stdout=stdout, stderr=stderr)
|
|
11
|
+
|
|
12
|
+
@patch("portacode.updater._running_in_virtualenv", return_value=True)
|
|
13
|
+
def test_venv_omits_user_flag_for_non_root(self, _mock_venv):
|
|
14
|
+
with patch("portacode.updater.sys.executable", "/opt/portacode-venv/bin/python"):
|
|
15
|
+
with patch("portacode.updater.os.geteuid", return_value=1000):
|
|
16
|
+
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
17
|
+
|
|
18
|
+
self.assertEqual(
|
|
19
|
+
cmd,
|
|
20
|
+
[
|
|
21
|
+
"/opt/portacode-venv/bin/python",
|
|
22
|
+
"-m",
|
|
23
|
+
"pip",
|
|
24
|
+
"install",
|
|
25
|
+
"--upgrade",
|
|
26
|
+
"portacode==1.4.34.dev7",
|
|
27
|
+
],
|
|
28
|
+
)
|
|
29
|
+
|
|
30
|
+
@patch("portacode.updater._running_in_virtualenv", return_value=False)
|
|
31
|
+
def test_non_root_outside_venv_uses_user_flag(self, _mock_venv):
|
|
32
|
+
with patch("portacode.updater.sys.executable", "/usr/bin/python3"):
|
|
33
|
+
with patch("portacode.updater.os.geteuid", return_value=1000):
|
|
34
|
+
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
35
|
+
|
|
36
|
+
self.assertEqual(cmd[-1], "--user")
|
|
37
|
+
self.assertEqual(cmd[:6], ["/usr/bin/python3", "-m", "pip", "install", "--upgrade", "portacode==1.4.34.dev7"])
|
|
38
|
+
|
|
39
|
+
@patch("portacode.updater._running_in_virtualenv", return_value=False)
|
|
40
|
+
@patch("portacode.updater.shutil.which", return_value="/usr/bin/sudo")
|
|
41
|
+
def test_root_with_sudo_user_outside_venv_switches_to_invoking_user(self, _mock_which, _mock_venv):
|
|
42
|
+
with patch("portacode.updater.sys.executable", "/usr/bin/python3"):
|
|
43
|
+
with patch("portacode.updater.os.geteuid", return_value=0):
|
|
44
|
+
with patch.dict("portacode.updater.os.environ", {"SUDO_USER": "ubuntu"}, clear=False):
|
|
45
|
+
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
46
|
+
|
|
47
|
+
self.assertEqual(cmd[:4], ["sudo", "-u", "ubuntu", "-H"])
|
|
48
|
+
self.assertEqual(cmd[-1], "--user")
|
|
49
|
+
|
|
50
|
+
@patch("portacode.updater.shutil.which", return_value="/usr/bin/sudo")
|
|
51
|
+
@patch("portacode.updater.os.geteuid", return_value=1000)
|
|
52
|
+
@patch("portacode.updater.subprocess.run")
|
|
53
|
+
def test_run_install_uses_sudo_n_on_permission_error(self, mock_run, _mock_euid, _mock_which):
|
|
54
|
+
mock_run.side_effect = [
|
|
55
|
+
self._cp(1, stderr="Permission denied: '/opt/portacode-venv/bin/portacode'"),
|
|
56
|
+
self._cp(0, stdout="ok"),
|
|
57
|
+
]
|
|
58
|
+
result = run_pip_install_command(
|
|
59
|
+
["/opt/portacode-venv/bin/python", "-m", "pip", "install", "--upgrade", "portacode==1.4.35.dev0"],
|
|
60
|
+
allow_sudo_fallback=True,
|
|
61
|
+
interactive_sudo=False,
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
self.assertEqual(result.returncode, 0)
|
|
65
|
+
self.assertEqual(mock_run.call_count, 2)
|
|
66
|
+
self.assertEqual(mock_run.call_args_list[1][0][0][:2], ["sudo", "-n"])
|
|
67
|
+
|
|
68
|
+
@patch("portacode.updater.shutil.which", return_value="/usr/bin/sudo")
|
|
69
|
+
@patch("portacode.updater.os.geteuid", return_value=1000)
|
|
70
|
+
@patch("portacode.updater.subprocess.run")
|
|
71
|
+
def test_run_install_falls_back_to_interactive_sudo_prompt(self, mock_run, _mock_euid, _mock_which):
|
|
72
|
+
mock_run.side_effect = [
|
|
73
|
+
self._cp(1, stderr="Permission denied"),
|
|
74
|
+
self._cp(1, stderr="sudo: a password is required"),
|
|
75
|
+
self._cp(0, stdout="ok"),
|
|
76
|
+
]
|
|
77
|
+
result = run_pip_install_command(
|
|
78
|
+
["/opt/portacode-venv/bin/python", "-m", "pip", "install", "--upgrade", "portacode==1.4.35.dev0"],
|
|
79
|
+
allow_sudo_fallback=True,
|
|
80
|
+
interactive_sudo=True,
|
|
81
|
+
)
|
|
82
|
+
|
|
83
|
+
self.assertEqual(result.returncode, 0)
|
|
84
|
+
self.assertEqual(mock_run.call_count, 3)
|
|
85
|
+
self.assertEqual(mock_run.call_args_list[1][0][0][:2], ["sudo", "-n"])
|
|
86
|
+
self.assertEqual(mock_run.call_args_list[2][0][0][0], "sudo")
|
|
87
|
+
|
|
88
|
+
@patch("portacode.updater.shutil.which", return_value="/usr/bin/sudo")
|
|
89
|
+
@patch("portacode.updater.os.geteuid", return_value=1000)
|
|
90
|
+
@patch("portacode.updater.subprocess.run")
|
|
91
|
+
def test_run_install_does_not_use_sudo_when_not_permission_issue(self, mock_run, _mock_euid, _mock_which):
|
|
92
|
+
mock_run.return_value = self._cp(1, stderr="No matching distribution found")
|
|
93
|
+
result = run_pip_install_command(
|
|
94
|
+
["/opt/portacode-venv/bin/python", "-m", "pip", "install", "--upgrade", "portacode==bad"],
|
|
95
|
+
allow_sudo_fallback=True,
|
|
96
|
+
interactive_sudo=True,
|
|
97
|
+
)
|
|
98
|
+
|
|
99
|
+
self.assertEqual(result.returncode, 1)
|
|
100
|
+
mock_run.assert_called_once()
|
|
@@ -5,6 +5,7 @@ from __future__ import annotations
|
|
|
5
5
|
import os
|
|
6
6
|
import sys
|
|
7
7
|
import shutil
|
|
8
|
+
import subprocess
|
|
8
9
|
from typing import List, Optional, Sequence
|
|
9
10
|
|
|
10
11
|
|
|
@@ -16,6 +17,53 @@ def _running_in_virtualenv() -> bool:
|
|
|
16
17
|
return sys.prefix != base_prefix
|
|
17
18
|
|
|
18
19
|
|
|
20
|
+
def _is_permission_error(output: str) -> bool:
|
|
21
|
+
text = (output or "").lower()
|
|
22
|
+
return "permission denied" in text or "errno 13" in text
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
def _sudo_requires_password(output: str) -> bool:
|
|
26
|
+
text = (output or "").lower()
|
|
27
|
+
return ("sudo:" in text and "password" in text) or "a password is required" in text
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
def run_pip_install_command(
|
|
31
|
+
cmd: Sequence[str],
|
|
32
|
+
*,
|
|
33
|
+
allow_sudo_fallback: bool = False,
|
|
34
|
+
interactive_sudo: bool = False,
|
|
35
|
+
) -> subprocess.CompletedProcess[str]:
|
|
36
|
+
"""Run a pip install command, with optional sudo fallback on permission errors."""
|
|
37
|
+
result = subprocess.run(list(cmd), capture_output=True, text=True)
|
|
38
|
+
if result.returncode == 0:
|
|
39
|
+
return result
|
|
40
|
+
|
|
41
|
+
if not allow_sudo_fallback:
|
|
42
|
+
return result
|
|
43
|
+
|
|
44
|
+
if os.geteuid() == 0 or not shutil.which("sudo"):
|
|
45
|
+
return result
|
|
46
|
+
|
|
47
|
+
combined = f"{result.stderr or ''}\n{result.stdout or ''}"
|
|
48
|
+
if not _is_permission_error(combined):
|
|
49
|
+
return result
|
|
50
|
+
|
|
51
|
+
sudo_non_interactive = subprocess.run(
|
|
52
|
+
["sudo", "-n", *cmd], capture_output=True, text=True
|
|
53
|
+
)
|
|
54
|
+
if sudo_non_interactive.returncode == 0:
|
|
55
|
+
return sudo_non_interactive
|
|
56
|
+
|
|
57
|
+
if not interactive_sudo:
|
|
58
|
+
return sudo_non_interactive
|
|
59
|
+
|
|
60
|
+
sudo_output = f"{sudo_non_interactive.stderr or ''}\n{sudo_non_interactive.stdout or ''}"
|
|
61
|
+
if _sudo_requires_password(sudo_output):
|
|
62
|
+
return subprocess.run(["sudo", *cmd], capture_output=True, text=True)
|
|
63
|
+
|
|
64
|
+
return sudo_non_interactive
|
|
65
|
+
|
|
66
|
+
|
|
19
67
|
def build_pip_install_command(
|
|
20
68
|
package: str = "portacode",
|
|
21
69
|
version: Optional[str] = None,
|
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
from unittest import TestCase
|
|
2
|
-
from unittest.mock import patch
|
|
3
|
-
|
|
4
|
-
from portacode.updater import build_pip_install_command
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
class UpdaterCommandTests(TestCase):
|
|
8
|
-
@patch("portacode.updater._running_in_virtualenv", return_value=True)
|
|
9
|
-
def test_venv_omits_user_flag_for_non_root(self, _mock_venv):
|
|
10
|
-
with patch("portacode.updater.sys.executable", "/opt/portacode-venv/bin/python"):
|
|
11
|
-
with patch("portacode.updater.os.geteuid", return_value=1000):
|
|
12
|
-
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
13
|
-
|
|
14
|
-
self.assertEqual(
|
|
15
|
-
cmd,
|
|
16
|
-
[
|
|
17
|
-
"/opt/portacode-venv/bin/python",
|
|
18
|
-
"-m",
|
|
19
|
-
"pip",
|
|
20
|
-
"install",
|
|
21
|
-
"--upgrade",
|
|
22
|
-
"portacode==1.4.34.dev7",
|
|
23
|
-
],
|
|
24
|
-
)
|
|
25
|
-
|
|
26
|
-
@patch("portacode.updater._running_in_virtualenv", return_value=False)
|
|
27
|
-
def test_non_root_outside_venv_uses_user_flag(self, _mock_venv):
|
|
28
|
-
with patch("portacode.updater.sys.executable", "/usr/bin/python3"):
|
|
29
|
-
with patch("portacode.updater.os.geteuid", return_value=1000):
|
|
30
|
-
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
31
|
-
|
|
32
|
-
self.assertEqual(cmd[-1], "--user")
|
|
33
|
-
self.assertEqual(cmd[:6], ["/usr/bin/python3", "-m", "pip", "install", "--upgrade", "portacode==1.4.34.dev7"])
|
|
34
|
-
|
|
35
|
-
@patch("portacode.updater._running_in_virtualenv", return_value=False)
|
|
36
|
-
@patch("portacode.updater.shutil.which", return_value="/usr/bin/sudo")
|
|
37
|
-
def test_root_with_sudo_user_outside_venv_switches_to_invoking_user(self, _mock_which, _mock_venv):
|
|
38
|
-
with patch("portacode.updater.sys.executable", "/usr/bin/python3"):
|
|
39
|
-
with patch("portacode.updater.os.geteuid", return_value=0):
|
|
40
|
-
with patch.dict("portacode.updater.os.environ", {"SUDO_USER": "ubuntu"}, clear=False):
|
|
41
|
-
cmd = build_pip_install_command(version="1.4.34.dev7")
|
|
42
|
-
|
|
43
|
-
self.assertEqual(cmd[:4], ["sudo", "-u", "ubuntu", "-H"])
|
|
44
|
-
self.assertEqual(cmd[-1], "--user")
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/cloudflared-domain-connect-containers-audit.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
{portacode-1.4.35.dev0 → portacode-1.4.35.dev2}/docs/homepage-dashboard-positioning-fixes.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|