From 2f325268c2733bbc1063b68424d3c71d9bc9b610 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Viktor=20Sandstr=C3=B6m?= Date: Fri, 31 May 2024 15:25:51 +0200 Subject: [PATCH 01/12] SS-1002 Feature/implement django forms (#203) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Nikita Churikov Co-authored-by: Johan Alfredéen This PR introduces complete rework of how Apps on serve are being created on the backend and changes the way app creation form look like on the frontend. In order to do that, orm models and forms needed to be changed. You can see on this page how to add new app type to serve https://scilifelab.atlassian.net/wiki/x/AQABrQ . This is a good way to understand what files do what. --- api/.github/dependabot.yml | 22 - api/.github/workflows/branch-name-check.yaml | 21 - api/.github/workflows/code-checks.yaml | 52 -- api/.gitignore | 129 --- api/LICENSE | 201 ----- api/MANIFEST.in | 1 - api/README.md | 26 - api/openapi/public_apps_api.py | 85 +- api/openapi/urls.py | 2 +- api/serializers.py | 38 +- api/setup.py | 33 - api/tests/test_openapi_public_apps.py | 24 +- api/urls.py | 10 +- api/views.py | 203 ++--- apps/.github/dependabot.yml | 22 - apps/.github/workflows/branch-name-check.yaml | 21 - apps/.github/workflows/code-checks.yaml | 52 -- apps/.gitignore | 129 --- apps/LICENSE | 201 ----- apps/MANIFEST.in | 2 - apps/README.md | 18 - apps/admin.py | 226 ++++-- apps/app_registry.py | 39 + apps/apps.py | 3 + apps/constants.py | 21 + apps/controller.py | 151 ---- apps/forms/__init__.py | 12 + apps/forms/base.py | 186 +++++ apps/forms/custom.py | 86 ++ apps/forms/custom_field.py | 17 + apps/forms/dash.py | 65 ++ apps/forms/filemanager.py | 46 ++ apps/forms/jupyter.py | 29 + apps/forms/netpolicy.py | 19 + apps/forms/rstudio.py | 27 + apps/forms/shiny.py | 95 +++ apps/forms/tissuumaps.py | 36 + apps/forms/volumes.py | 24 + apps/forms/vscode.py | 27 + apps/generate_form.py | 280 ------- apps/helpers.py | 326 ++++---- apps/migrations/0001_initial.py | 211 +++++ apps/migrations/0002_initial.py | 526 ++++++++++++ apps/models.py | 223 ----- apps/models/__init__.py | 4 + apps/models/app_types/__init__.py | 10 + apps/models/app_types/custom.py | 50 ++ apps/models/app_types/dash.py | 40 + apps/models/app_types/filemanager.py | 40 + apps/models/app_types/jupyter.py | 41 + apps/models/app_types/netpolicy.py | 36 + apps/models/app_types/rstudio.py | 42 + apps/models/app_types/shiny.py | 50 ++ apps/models/app_types/tissuumaps.py | 50 ++ apps/models/app_types/volume.py | 31 + apps/models/app_types/vscode.py | 38 + apps/models/base/__init__.py | 7 + apps/models/base/app_categories.py | 14 + apps/models/base/app_status.py | 15 + apps/models/base/app_template.py | 43 + apps/models/base/base.py | 161 ++++ apps/models/base/logs_enabled_mixin.py | 10 + apps/models/base/social_mixin.py | 13 + apps/models/base/subdomain.py | 20 + apps/serialize.py | 369 --------- apps/setup.py | 35 - apps/signals.py | 71 ++ apps/tasks.py | 602 +++----------- apps/tests/test_app_instance.py | 111 ++- apps/tests/test_app_instance_manager.py | 163 ++-- apps/tests/test_app_settings_view.py | 36 +- apps/tests/test_app_view_forbidden.py | 147 +--- apps/tests/test_create_app_view.py | 42 +- apps/tests/test_delete_app_view.py | 31 +- apps/tests/test_forms.py | 192 +++++ apps/tests/test_generate_form.py | 305 ------- apps/tests/test_get_status_view.py | 11 +- apps/tests/test_subdomain_candidate.py | 62 ++ apps/tests/test_update_status_handler.py | 85 +- .../types_}/__init__.py | 0 apps/types_/app_registry.py | 35 + apps/types_/app_types.py | 14 + apps/types_/subdomain.py | 49 ++ apps/urls.py | 20 +- apps/views.py | 760 ++++-------------- collections_module/admin.py | 15 - collections_module/apps.py | 6 - collections_module/migrations/0001_initial.py | 37 - .../0002_collection_zenodo_community_id.py | 17 - collections_module/models.py | 28 - collections_module/tests.py | 0 collections_module/urls.py | 10 - collections_module/views.py | 36 - .../management/commands/create_locust_apps.py | 5 +- .../management/commands/install_fixtures.py | 6 +- common/migrations/0001_initial.py | 31 +- .../migrations/0002_emailverificationtable.py | 26 - common/migrations/0003_fixtureversion.py | 20 - .../migrations/0004_userprofile_deleted_on.py | 17 - common/migrations/0005_maintenancemode.py | 21 - .../templatetags}/__init__.py | 0 .../templatetags/can_create_app.py | 5 +- .../templatetags/get_dict_key.py | 0 .../templatetags/get_range.py | 0 .../templatetags/get_setting.py | 0 .../templatetags/get_version.py | 0 .../templatetags/is_login_signup_disabled.py | 0 customtags/__init__.py | 0 customtags/migrations/__init__.py | 0 customtags/templatetags/__init__.py | 0 .../setup-scripts/seed_collections_user.py | 56 +- cypress/e2e/setup-scripts/seed_contributor.py | 11 +- cypress/e2e/setup-scripts/seed_superuser.py | 85 +- cypress/e2e/ui-tests/test-collections.cy.js | 2 +- cypress/e2e/ui-tests/test-deploy-app.cy.js | 231 +++--- .../test-project-as-contributor.cy.js | 28 +- .../test-superuser-functionality.cy.js | 50 +- fixtures/apps_fixtures.json | 454 +---------- fixtures/periodic_tasks_fixtures.json | 57 +- fixtures/projects_templates.json | 59 +- models/.github/dependabot.yml | 22 - .../.github/workflows/branch-name-check.yaml | 21 - models/.github/workflows/code-checks.yaml | 52 -- models/.gitignore | 129 --- models/LICENSE | 201 ----- models/MANIFEST.in | 2 - models/README.md | 18 - .../migrations}/0001_initial.py | 56 +- models/models.py | 2 +- models/setup.py | 33 - models/tests.py | 418 ---------- models/views.py | 10 +- monitor/.github/dependabot.yml | 22 - .../.github/workflows/branch-name-check.yaml | 21 - monitor/.github/workflows/code-checks.yaml | 52 -- monitor/.gitignore | 129 --- monitor/LICENSE | 201 ----- monitor/MANIFEST.in | 2 - monitor/README.md | 18 - monitor/__init__.py | 0 monitor/admin.py | 0 monitor/apps.py | 7 - monitor/dash_demo.py | 201 ----- monitor/helpers.py | 178 ---- monitor/migrations/__init__.py | 0 monitor/models.py | 0 monitor/setup.py | 27 - monitor/templates/monitor_new.html | 33 - monitor/templates/monitor_overview.html | 10 - monitor/tests.py | 0 monitor/urls.py | 18 - monitor/views.py | 267 ------ news/LICENSE | 201 ----- news/MANIFEST.in | 2 - news/README.md | 18 - news/__init__.py | 0 news/admin.py | 5 - news/apps.py | 7 - news/migrations/0001_initial.py | 20 - news/migrations/__init__.py | 0 news/models.py | 18 - news/setup.py | 28 - news/tasks.py | 0 news/tests.py | 0 news/urls.py | 11 - news/views.py | 14 - poetry.lock | 192 +++-- portal/.github/dependabot.yml | 22 - .../.github/workflows/branch-name-check.yaml | 21 - portal/.github/workflows/code-checks.yaml | 52 -- portal/.gitignore | 129 --- portal/LICENSE | 201 ----- portal/MANIFEST.in | 2 - portal/README.md | 18 - portal/admin.py | 14 +- portal/migrations/0001_initial.py | 74 ++ portal/models.py | 43 +- portal/setup.py | 28 - portal/urls.py | 3 + portal/views.py | 103 ++- projects/.github/dependabot.yml | 22 - .../.github/workflows/branch-name-check.yaml | 21 - projects/.github/workflows/code-checks.yaml | 52 -- projects/.gitignore | 129 --- projects/LICENSE | 201 ----- projects/MANIFEST.in | 2 - projects/README.md | 18 - projects/admin.py | 17 +- projects/apps.py | 2 +- projects/helpers.py | 25 - .../migrations}/0001_initial.py | 230 ++---- projects/models.py | 128 +-- projects/setup.py | 30 - projects/static/img/logo.png | Bin 2226 -> 0 bytes projects/tasks.py | 289 ++++--- projects/tests/test_create_mlflow.py | 52 -- projects/tests/test_project.py | 8 +- projects/tests/test_project_view.py | 30 - projects/urls.py | 6 - projects/views.py | 166 +--- pyproject.toml | 16 +- scripts/app_instance_permissions.py | 23 +- static/css/serve-elements.css | 21 + studio/migrations/__init__.py | 0 studio/migrations/apps/0001_initial.py | 128 --- studio/migrations/apps/0002_initial.py | 83 -- ...03_appinstance_note_on_linkonly_privacy.py | 17 - ...ve_appinstance_note_on_linkonly_privacy.py | 16 - ...05_appinstance_note_on_linkonly_privacy.py | 17 - .../apps/0006_appinstance_collections.py | 18 - .../apps/0007_appinstance_source_code_url.py | 17 - studio/migrations/apps/__init__.py | 0 studio/migrations/models/__init__.py | 0 studio/migrations/monitor/__init__.py | 0 studio/migrations/portal/0001_initial.py | 38 - .../portal/0002_publishedmodel_collections.py | 20 - studio/migrations/portal/__init__.py | 0 .../projects/0002_project_project_template.py | 20 - studio/migrations/projects/__init__.py | 0 studio/settings.py | 29 +- studio/tests.py | 8 +- studio/urls.py | 6 - studio/views.py | 50 +- templates/apps/create.html | 511 ------------ templates/apps/create_base.html | 186 +++++ templates/apps/create_view.html | 19 + templates/apps/custom_field.html | 25 + templates/apps/logs.html | 27 +- templates/apps/update.html | 601 -------------- templates/breadcrumbs/bc_app_create.html | 10 + templates/breadcrumbs/bc_logs.html | 5 + templates/breadcrumbs/bc_project_create.html | 6 + .../breadcrumbs/bc_project_overview.html | 1 + templates/breadcrumbs/breadcrumb_base.html | 10 + templates/collections/collection.html | 2 +- templates/common/app_card copy.html | 119 +++ templates/common/app_card.html | 8 +- templates/common/footer.html | 2 +- templates/portal/home.html | 6 +- templates/projects/categories/develop.html | 13 + .../categories/manage_files copy.html | 61 ++ .../projects/categories/manage_files.html | 13 + templates/projects/categories/models.html | 137 ++++ templates/projects/categories/serve.html | 9 + templates/projects/overview.html | 581 +------------ .../partials/app_instances_table.html | 114 +++ .../projects/partials/app_templates.html | 68 ++ .../projects/partials/category_card_base.html | 28 + .../partials/project_description.html | 11 + .../projects/partials/project_header.html | 7 + templates/projects/partials/scripts.html | 118 +++ templates/projects/project_create.html | 7 +- templates/projects/settings.html | 69 -- 253 files changed, 5652 insertions(+), 11588 deletions(-) delete mode 100644 api/.github/dependabot.yml delete mode 100644 api/.github/workflows/branch-name-check.yaml delete mode 100644 api/.github/workflows/code-checks.yaml delete mode 100644 api/.gitignore delete mode 100644 api/LICENSE delete mode 100644 api/MANIFEST.in delete mode 100644 api/README.md delete mode 100644 api/setup.py delete mode 100644 apps/.github/dependabot.yml delete mode 100644 apps/.github/workflows/branch-name-check.yaml delete mode 100644 apps/.github/workflows/code-checks.yaml delete mode 100644 apps/.gitignore delete mode 100644 apps/LICENSE delete mode 100644 apps/MANIFEST.in delete mode 100644 apps/README.md create mode 100644 apps/app_registry.py create mode 100644 apps/constants.py delete mode 100644 apps/controller.py create mode 100644 apps/forms/__init__.py create mode 100644 apps/forms/base.py create mode 100644 apps/forms/custom.py create mode 100644 apps/forms/custom_field.py create mode 100644 apps/forms/dash.py create mode 100644 apps/forms/filemanager.py create mode 100644 apps/forms/jupyter.py create mode 100644 apps/forms/netpolicy.py create mode 100644 apps/forms/rstudio.py create mode 100644 apps/forms/shiny.py create mode 100644 apps/forms/tissuumaps.py create mode 100644 apps/forms/volumes.py create mode 100644 apps/forms/vscode.py delete mode 100644 apps/generate_form.py create mode 100644 apps/migrations/0001_initial.py create mode 100644 apps/migrations/0002_initial.py delete mode 100644 apps/models.py create mode 100644 apps/models/__init__.py create mode 100644 apps/models/app_types/__init__.py create mode 100644 apps/models/app_types/custom.py create mode 100644 apps/models/app_types/dash.py create mode 100644 apps/models/app_types/filemanager.py create mode 100644 apps/models/app_types/jupyter.py create mode 100644 apps/models/app_types/netpolicy.py create mode 100644 apps/models/app_types/rstudio.py create mode 100644 apps/models/app_types/shiny.py create mode 100644 apps/models/app_types/tissuumaps.py create mode 100644 apps/models/app_types/volume.py create mode 100644 apps/models/app_types/vscode.py create mode 100644 apps/models/base/__init__.py create mode 100644 apps/models/base/app_categories.py create mode 100644 apps/models/base/app_status.py create mode 100644 apps/models/base/app_template.py create mode 100644 apps/models/base/base.py create mode 100644 apps/models/base/logs_enabled_mixin.py create mode 100644 apps/models/base/social_mixin.py create mode 100644 apps/models/base/subdomain.py delete mode 100644 apps/serialize.py delete mode 100644 apps/setup.py create mode 100644 apps/signals.py create mode 100644 apps/tests/test_forms.py delete mode 100644 apps/tests/test_generate_form.py create mode 100644 apps/tests/test_subdomain_candidate.py rename {collections_module => apps/types_}/__init__.py (100%) create mode 100644 apps/types_/app_registry.py create mode 100644 apps/types_/app_types.py create mode 100644 apps/types_/subdomain.py delete mode 100644 collections_module/admin.py delete mode 100644 collections_module/apps.py delete mode 100644 collections_module/migrations/0001_initial.py delete mode 100644 collections_module/migrations/0002_collection_zenodo_community_id.py delete mode 100644 collections_module/models.py delete mode 100644 collections_module/tests.py delete mode 100644 collections_module/urls.py delete mode 100644 collections_module/views.py delete mode 100644 common/migrations/0002_emailverificationtable.py delete mode 100644 common/migrations/0003_fixtureversion.py delete mode 100644 common/migrations/0004_userprofile_deleted_on.py delete mode 100644 common/migrations/0005_maintenancemode.py rename {collections_module/migrations => common/templatetags}/__init__.py (100%) rename {customtags => common}/templatetags/can_create_app.py (62%) rename {customtags => common}/templatetags/get_dict_key.py (100%) rename {customtags => common}/templatetags/get_range.py (100%) rename {customtags => common}/templatetags/get_setting.py (100%) rename {customtags => common}/templatetags/get_version.py (100%) rename {customtags => common}/templatetags/is_login_signup_disabled.py (100%) delete mode 100644 customtags/__init__.py delete mode 100644 customtags/migrations/__init__.py delete mode 100644 customtags/templatetags/__init__.py delete mode 100644 models/.github/dependabot.yml delete mode 100644 models/.github/workflows/branch-name-check.yaml delete mode 100644 models/.github/workflows/code-checks.yaml delete mode 100644 models/.gitignore delete mode 100644 models/LICENSE delete mode 100644 models/MANIFEST.in delete mode 100644 models/README.md rename {studio/migrations/models => models/migrations}/0001_initial.py (96%) delete mode 100644 models/setup.py delete mode 100644 monitor/.github/dependabot.yml delete mode 100644 monitor/.github/workflows/branch-name-check.yaml delete mode 100644 monitor/.github/workflows/code-checks.yaml delete mode 100644 monitor/.gitignore delete mode 100644 monitor/LICENSE delete mode 100644 monitor/MANIFEST.in delete mode 100644 monitor/README.md delete mode 100644 monitor/__init__.py delete mode 100644 monitor/admin.py delete mode 100644 monitor/apps.py delete mode 100644 monitor/dash_demo.py delete mode 100644 monitor/helpers.py delete mode 100644 monitor/migrations/__init__.py delete mode 100644 monitor/models.py delete mode 100644 monitor/setup.py delete mode 100644 monitor/templates/monitor_new.html delete mode 100644 monitor/templates/monitor_overview.html delete mode 100644 monitor/tests.py delete mode 100644 monitor/urls.py delete mode 100644 monitor/views.py delete mode 100644 news/LICENSE delete mode 100644 news/MANIFEST.in delete mode 100644 news/README.md delete mode 100644 news/__init__.py delete mode 100644 news/admin.py delete mode 100644 news/apps.py delete mode 100644 news/migrations/0001_initial.py delete mode 100644 news/migrations/__init__.py delete mode 100644 news/models.py delete mode 100644 news/setup.py delete mode 100644 news/tasks.py delete mode 100644 news/tests.py delete mode 100644 news/urls.py delete mode 100644 news/views.py delete mode 100644 portal/.github/dependabot.yml delete mode 100644 portal/.github/workflows/branch-name-check.yaml delete mode 100644 portal/.github/workflows/code-checks.yaml delete mode 100644 portal/.gitignore delete mode 100644 portal/LICENSE delete mode 100644 portal/MANIFEST.in delete mode 100644 portal/README.md create mode 100644 portal/migrations/0001_initial.py delete mode 100644 portal/setup.py delete mode 100644 projects/.github/dependabot.yml delete mode 100644 projects/.github/workflows/branch-name-check.yaml delete mode 100644 projects/.github/workflows/code-checks.yaml delete mode 100644 projects/.gitignore delete mode 100644 projects/LICENSE delete mode 100644 projects/MANIFEST.in delete mode 100644 projects/README.md delete mode 100644 projects/helpers.py rename {studio/migrations/projects => projects/migrations}/0001_initial.py (56%) delete mode 100644 projects/setup.py delete mode 100644 projects/static/img/logo.png delete mode 100644 projects/tests/test_create_mlflow.py delete mode 100644 studio/migrations/__init__.py delete mode 100644 studio/migrations/apps/0001_initial.py delete mode 100644 studio/migrations/apps/0002_initial.py delete mode 100644 studio/migrations/apps/0003_appinstance_note_on_linkonly_privacy.py delete mode 100644 studio/migrations/apps/0004_remove_appinstance_note_on_linkonly_privacy.py delete mode 100644 studio/migrations/apps/0005_appinstance_note_on_linkonly_privacy.py delete mode 100644 studio/migrations/apps/0006_appinstance_collections.py delete mode 100644 studio/migrations/apps/0007_appinstance_source_code_url.py delete mode 100644 studio/migrations/apps/__init__.py delete mode 100644 studio/migrations/models/__init__.py delete mode 100644 studio/migrations/monitor/__init__.py delete mode 100644 studio/migrations/portal/0001_initial.py delete mode 100644 studio/migrations/portal/0002_publishedmodel_collections.py delete mode 100644 studio/migrations/portal/__init__.py delete mode 100644 studio/migrations/projects/0002_project_project_template.py delete mode 100644 studio/migrations/projects/__init__.py delete mode 100644 templates/apps/create.html create mode 100644 templates/apps/create_base.html create mode 100644 templates/apps/create_view.html create mode 100644 templates/apps/custom_field.html delete mode 100644 templates/apps/update.html create mode 100644 templates/breadcrumbs/bc_app_create.html create mode 100644 templates/breadcrumbs/bc_logs.html create mode 100644 templates/breadcrumbs/bc_project_create.html create mode 100644 templates/breadcrumbs/bc_project_overview.html create mode 100644 templates/breadcrumbs/breadcrumb_base.html create mode 100644 templates/common/app_card copy.html create mode 100644 templates/projects/categories/develop.html create mode 100644 templates/projects/categories/manage_files copy.html create mode 100644 templates/projects/categories/manage_files.html create mode 100644 templates/projects/categories/models.html create mode 100644 templates/projects/categories/serve.html create mode 100644 templates/projects/partials/app_instances_table.html create mode 100644 templates/projects/partials/app_templates.html create mode 100644 templates/projects/partials/category_card_base.html create mode 100644 templates/projects/partials/project_description.html create mode 100644 templates/projects/partials/project_header.html create mode 100644 templates/projects/partials/scripts.html diff --git a/api/.github/dependabot.yml b/api/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/api/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/api/.github/workflows/branch-name-check.yaml b/api/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 77324ca95..000000000 --- a/api/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v3 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/api/.github/workflows/code-checks.yaml b/api/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/api/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/api/.gitignore b/api/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/api/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/api/LICENSE b/api/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/api/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/api/MANIFEST.in b/api/MANIFEST.in deleted file mode 100644 index bb3ec5f0d..000000000 --- a/api/MANIFEST.in +++ /dev/null @@ -1 +0,0 @@ -include README.md diff --git a/api/README.md b/api/README.md deleted file mode 100644 index 4ca6ed406..000000000 --- a/api/README.md +++ /dev/null @@ -1,26 +0,0 @@ -# studio-api - -Restful-API Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "rest_framework.authtoken", - "rest_framework", - "api" -] - -REST_FRAMEWORK = { - "DEFAULT_AUTHENTICATION_CLASSES": [ - "rest_framework.authentication.TokenAuthentication" - ], -} -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/api/openapi/public_apps_api.py b/api/openapi/public_apps_api.py index fc02b9e9b..7e198ffb2 100644 --- a/api/openapi/public_apps_api.py +++ b/api/openapi/public_apps_api.py @@ -1,10 +1,12 @@ +from django.core.exceptions import FieldError from django.db.models import Q from django.http import JsonResponse from django.shortcuts import get_object_or_404 from rest_framework import viewsets from rest_framework.exceptions import NotFound -from apps.models import AppInstance, Apps +from apps.app_registry import APP_REGISTRY +from apps.models import Apps, AppStatus from studio.utils import get_logger logger = get_logger(__name__) @@ -23,38 +25,77 @@ def list(self, request): """ logger.info("PublicAppsAPI. Entered list method.") logger.info("Requested API version %s", request.version) + list_apps = [] - queryset = ( - AppInstance.objects.filter(~Q(state="Deleted"), access="public") - .order_by("-updated_on")[:8] - .values("id", "name", "app_id", "table_field", "description", "updated_on") - ) + # TODO: MAKE SURE THAT THIS IS FILTERED BASED ON ACCESS + for model_class in APP_REGISTRY.iter_orm_models(): + # Loop over all models, and check if they have the access and description field + if hasattr(model_class, "description") and hasattr(model_class, "access"): + queryset = ( + model_class.objects.filter(~Q(app_status__status="Deleted"), access="public") + .order_by("-updated_on")[:8] + .values("id", "name", "app_id", "url", "description", "updated_on", "app_status") + ) + list_apps.extend(list(queryset)) - list_apps = list(queryset) for app in list_apps: - add_data = Apps.objects.get(id=app["app_id"]) - app["app_type"] = add_data.name + app["app_type"] = Apps.objects.get(id=app["app_id"]).name + app["app_status"] = AppStatus.objects.get(pk=app["app_status"]).status + + # Add the previous url key located at app.table_field.url to support clients using the previous schema + app["table_field"] = {"url": app["url"]} + data = {"data": list_apps} logger.info("LIST: %s", data) return JsonResponse(data) - def retrieve(self, request, pk=None): + def retrieve(self, request, app_slug=None, pk=None): """ This endpoint retrieves a single public app instance. :returns dict: A dict of app information. """ logger.info("PublicAppsAPI. Entered retrieve method with pk = %s", pk) logger.info("Requested API version %s", self.request.version) - queryset = AppInstance.objects.all().values( - "id", "name", "app_id", "table_field", "description", "updated_on", "access", "state" - ) - app = get_object_or_404(queryset, pk=pk) - if app["state"] == "Deleted": - raise NotFound("this app has been deleted") - if app["access"] != "public": - raise NotFound() - - add_data = Apps.objects.get(id=app["app_id"]) - app["app_type"] = add_data.name - data = {"app": app} + + model_class = APP_REGISTRY.get_orm_model(app_slug) + + if model_class is None: + logger.error("App slug has no corresponding model class") + raise NotFound("Invalid app slug") + + try: + queryset = model_class.objects.all().values( + "id", "name", "app_id", "url", "description", "updated_on", "access", "app_status" + ) + logger.info("Queryset: %s", queryset) + except FieldError as e: + message = f"Error in field: {e}" + logger.error(f"App type is not available in public view: {message}") + raise NotFound("App type is not available in public view") + + app_instance = get_object_or_404(queryset, pk=pk) + if app_instance is None: + logger.error("App instance is not available") + raise NotFound("App instance is not available") + + app_status_pk = app_instance.get("app_status", None) + logger.info("DID WE GET HERE?!") + if app_status_pk is None: + raise NotFound("App status is not available") + + app_status = AppStatus.objects.get(pk=app_status_pk) + if app_status.status == "Deleted": + logger.error("This app has been deleted") + raise NotFound("This app has been deleted") + + if app_instance.get("access", False) != "public": + logger.error("This app is non-existent or not public") + raise NotFound("This app is non-existent or not public") + + app_instance["app_status"] = app_status.status + + add_data = Apps.objects.get(id=app_instance["app_id"]) + app_instance["app_type"] = add_data.name + data = {"app": app_instance} + logger.info("API call successful") return JsonResponse(data) diff --git a/api/openapi/urls.py b/api/openapi/urls.py index 9c6cf6f16..18cd55ca3 100644 --- a/api/openapi/urls.py +++ b/api/openapi/urls.py @@ -21,7 +21,7 @@ path("api-info", APIInfo.as_view({"get": "get_api_info"})), # The Apps API path("public-apps", PublicAppsAPI.as_view({"get": "list"})), - path("public-apps/", PublicAppsAPI.as_view({"get": "retrieve"})), + path("public-apps//", PublicAppsAPI.as_view({"get": "retrieve"})), # Supplementary lookups API path( "lookups/universities", diff --git a/api/serializers.py b/api/serializers.py index 31362ca27..680175ca3 100644 --- a/api/serializers.py +++ b/api/serializers.py @@ -1,17 +1,9 @@ from django.contrib.auth.models import User from rest_framework.serializers import ModelSerializer -from apps.models import AppCategories, AppInstance, Apps, AppStatus +from apps.models import AppCategories, Apps, AppStatus, BaseAppInstance from models.models import Metadata, Model, ModelLog, ObjectType -from projects.models import ( - S3, - Environment, - Flavor, - MLFlow, - Project, - ProjectTemplate, - ReleaseName, -) +from projects.models import Environment, Flavor, Project, ProjectTemplate class MLModelSerializer(ModelSerializer): @@ -72,23 +64,7 @@ class Meta: ) -class S3serializer(ModelSerializer): - class Meta: - model = S3 - fields = ("name", "access_key", "secret_key", "host", "region") - - -class MLflowSerializer(ModelSerializer): - s3 = S3serializer() - - class Meta: - model = MLFlow - fields = ("name", "mlflow_url", "s3") - - class ProjectSerializer(ModelSerializer): - s3storage = S3serializer() - class Meta: model = Project @@ -130,7 +106,7 @@ class AppInstanceSerializer(ModelSerializer): status = AppStatusSerializer(many=True) class Meta: - model = AppInstance + model = BaseAppInstance fields = ("id", "name", "app", "table_field", "state", "status") @@ -154,14 +130,6 @@ class Meta: fields = "__all__" -class ReleaseNameSerializer(ModelSerializer): - app = AppInstanceSerializer() - - class Meta: - model = ReleaseName - fields = "__all__" - - class ProjectTemplateSerializer(ModelSerializer): class Meta: model = ProjectTemplate diff --git a/api/setup.py b/api/setup.py deleted file mode 100644 index 97138c6e2..000000000 --- a/api/setup.py +++ /dev/null @@ -1,33 +0,0 @@ -from setuptools import setup - -setup( - name="studio-api", - version="0.0.1", - description="""Django app for handling REST-API in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["api"], - package_dir={"api": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.1.7", - "requests==2.28.1", - "django-guardian==2.4.0", - "django-tagulous==1.3.3", - "django-filter==22.1", - "drf-nested-routers==0.93.4", - "minio==7.0.2", - "s3fs==2022.1.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/api/tests/test_openapi_public_apps.py b/api/tests/test_openapi_public_apps.py index aa034549f..11cdcef11 100644 --- a/api/tests/test_openapi_public_apps.py +++ b/api/tests/test_openapi_public_apps.py @@ -6,7 +6,7 @@ from rest_framework import status from rest_framework.test import APITestCase -from apps.models import AppInstance, Apps +from apps.models import Apps, AppStatus, CustomAppInstance, Subdomain from projects.models import Project from studio.utils import get_logger @@ -27,18 +27,22 @@ class PublicAppsApiTests(APITestCase): def setUpTestData(cls): cls.user = User.objects.create_user(test_user["username"], test_user["email"], test_user["password"]) cls.project = Project.objects.create_project(name="test-perm", owner=cls.user, description="") - cls.app = Apps.objects.create(name="Some App", slug="some-app") + cls.app = Apps.objects.create(name="Some App", slug="customapp") - cls.app_instance = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Running") + cls.app_instance = CustomAppInstance.objects.create( access="public", owner=cls.user, name="test_app_instance_public", description="My app description", app=cls.app, project=cls.project, - parameters={ + k8s_values={ "environment": {"pk": ""}, }, + subdomain=subdomain, + app_status=app_status, ) def test_public_apps_list(self): @@ -64,10 +68,12 @@ def test_public_apps_list(self): def test_public_apps_single_app(self): """Tests the API resource public-apps get single object""" id = str(self.app_instance.id) - url = os.path.join(self.BASE_API_URL, "public-apps/", id) + app_slug = self.app_instance.app.slug + url = os.path.join(self.BASE_API_URL, "public-apps", app_slug, id) response = self.client.get(url, format="json") self.assertEqual(response.status_code, status.HTTP_200_OK) + logger.info(response.content) actual = json.loads(response.content)["app"] logger.info(type(actual)) @@ -84,6 +90,12 @@ def test_public_apps_single_app(self): def test_public_apps_single_app_notfound(self): """Tests the API resource public-apps get single object for a non-existing id""" id = "-1" - url = os.path.join(self.BASE_API_URL, "public-apps/", id) + app_slug = self.app_instance.app.slug + url = os.path.join(self.BASE_API_URL, "public-apps", app_slug, id) + response = self.client.get(url, format="json") + self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND) + + app_slug = "invalid_app_slug" + url = os.path.join(self.BASE_API_URL, "public-apps", app_slug, id) response = self.client.get(url, format="json") self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND) diff --git a/api/urls.py b/api/urls.py index 7046ec5a0..47791c628 100644 --- a/api/urls.py +++ b/api/urls.py @@ -12,15 +12,14 @@ FlavorsList, MembersList, MetadataList, - MLflowList, ModelList, ModelLogList, ObjectTypeList, ProjectList, ProjectTemplateList, - ReleaseNameList, ResourceList, - S3List, + get_subdomain_is_available, + get_subdomain_is_valid, update_app_status, ) @@ -40,9 +39,6 @@ models_router.register(r"appinstances", AppInstanceList, basename="appinstances") models_router.register(r"flavors", FlavorsList, basename="flavors") models_router.register(r"environments", EnvironmentList, basename="environment") -models_router.register(r"s3", S3List, basename="s3") -models_router.register(r"mlflow", MLflowList, basename="mlflow") -models_router.register(r"releasenames", ReleaseNameList, basename="releasenames") models_router.register(r"modellogs", ModelLogList, basename="modellog") models_router.register(r"metadata", MetadataList, basename="metadata") models_router.register(r"apps", AppList, basename="apps") @@ -57,4 +53,6 @@ path("token-auth/", CustomAuthToken.as_view(), name="api_token_auth"), path("settings/", get_studio_settings), path("app-status/", update_app_status), + path("app-subdomain/validate/", get_subdomain_is_valid), + path("app-subdomain/is-available/", get_subdomain_is_available), ] diff --git a/api/views.py b/api/views.py index f753e886d..e793c9176 100644 --- a/api/views.py +++ b/api/views.py @@ -5,10 +5,9 @@ import pytz from django.conf import settings from django.contrib.auth.models import User -from django.core.exceptions import ObjectDoesNotExist +from django.core.exceptions import ObjectDoesNotExist, ValidationError from django.db.models import Q -from django.http import HttpResponse -from django.utils.text import slugify +from django.http import HttpRequest, HttpResponse from django_filters.rest_framework import DjangoFilterBackend from rest_framework import generics from rest_framework.authtoken.models import Token @@ -25,19 +24,12 @@ from rest_framework.viewsets import GenericViewSet from apps.helpers import HandleUpdateStatusResponseCode, handle_update_status_request -from apps.models import AppCategories, AppInstance, Apps, AppStatus +from apps.models import AppCategories, Apps, BaseAppInstance from apps.tasks import delete_resource +from apps.types_.subdomain import SubdomainCandidateName from models.models import ObjectType from portal.models import PublishedModel -from projects.models import ( - S3, - Environment, - Flavor, - MLFlow, - ProjectLog, - ProjectTemplate, - ReleaseName, -) +from projects.models import Environment, Flavor, ProjectLog, ProjectTemplate from projects.tasks import create_resources_from_template, delete_project_apps from studio.utils import get_logger @@ -49,7 +41,6 @@ FlavorsSerializer, Metadata, MetadataSerializer, - MLflowSerializer, MLModelSerializer, Model, ModelLog, @@ -58,8 +49,6 @@ Project, ProjectSerializer, ProjectTemplateSerializer, - ReleaseNameSerializer, - S3serializer, UserSerializer, ) @@ -491,7 +480,7 @@ class AppInstanceList( filterset_fields = ["id", "name", "app__category"] def get_queryset(self): - return AppInstance.objects.filter(~Q(state="Deleted"), project__pk=self.kwargs["project_pk"]) + return BaseAppInstance.objects.filter(~Q(state="Deleted"), project__pk=self.kwargs["project_pk"]) def create(self, request, *args, **kwargs): project = Project.objects.get(id=self.kwargs["project_pk"]) @@ -595,103 +584,6 @@ def destroy(self, request, *args, **kwargs): return HttpResponse("Deleted object.", status=200) -class S3List( - GenericViewSet, - CreateModelMixin, - RetrieveModelMixin, - UpdateModelMixin, - ListModelMixin, -): - permission_classes = ( - IsAuthenticated, - ProjectPermission, - ) - serializer_class = S3serializer - filter_backends = [DjangoFilterBackend] - filterset_fields = ["id", "name", "host", "region"] - - def get_queryset(self): - return S3.objects.filter(project__pk=self.kwargs["project_pk"]) - - def destroy(self, request, *args, **kwargs): - try: - obj = self.get_object() - except Exception as e: - logger.error(e, exc_info=True) - return HttpResponse("No such object.", status=400) - obj.delete() - return HttpResponse("Deleted object.", status=200) - - -class MLflowList( - GenericViewSet, - CreateModelMixin, - RetrieveModelMixin, - UpdateModelMixin, - ListModelMixin, -): - permission_classes = ( - IsAuthenticated, - ProjectPermission, - ) - serializer_class = MLflowSerializer - filter_backends = [DjangoFilterBackend] - filterset_fields = ["id", "name"] - - def get_queryset(self): - return MLFlow.objects.filter(project__pk=self.kwargs["project_pk"]) - - def destroy(self, request, *args, **kwargs): - try: - obj = self.get_object() - except Exception as e: - logger.error(e, exc_info=True) - return HttpResponse("No such object.", status=400) - obj.delete() - return HttpResponse("Deleted object.", status=200) - - -class ReleaseNameList( - GenericViewSet, - CreateModelMixin, - RetrieveModelMixin, - UpdateModelMixin, - ListModelMixin, -): - permission_classes = ( - IsAuthenticated, - ProjectPermission, - ) - serializer_class = ReleaseNameSerializer - filter_backends = [DjangoFilterBackend] - filterset_fields = ["id", "name", "project"] - - def get_queryset(self): - return ReleaseName.objects.filter(project__pk=self.kwargs["project_pk"]) - - def create(self, request, *args, **kwargs): - name = slugify(request.data["name"]) - project = Project.objects.get(id=self.kwargs["project_pk"]) - if ReleaseName.objects.filter(name=name).exists(): - if project.status != "archived": - logger.info("ReleaseName already in use.") - return HttpResponse("Release name already in use.", status=200) - status = "active" - - rn = ReleaseName(name=name, status=status, project=project) - rn.save() - return HttpResponse("Created release name {}.".format(name), status=200) - - def destroy(self, request, *args, **kwargs): - try: - obj = self.get_object() - except Exception as e: - logger.error(e, exc_info=True) - return HttpResponse("No such object.", status=400) - obj.delete() - return HttpResponse("Deleted object.", status=200) - - class AppList( generics.ListAPIView, GenericViewSet, @@ -834,6 +726,85 @@ def create(self, request, *args, **kwargs): return HttpResponse("Created new template: {}.".format(name), status=200) +@api_view(["GET"]) +@permission_classes( + ( + # IsAuthenticated, + ) +) +def get_subdomain_is_valid(request: HttpRequest) -> HttpResponse: + """ + Implementation of the API method at endpoint /api/app-subdomain/validate/ + Supports the GET verb. + + The service contract for the GET action is as follows: + :param str subdomainText: The subdomain text to validate. + :returns: An http status code and dict containing {"isValid": bool, "message": str} + + Example request: /api/app-subdomain/validate/?subdomainText=my-subdomain + """ + subdomain_text = request.GET.get("subdomainText") + + if subdomain_text is None: + return Response("Invalid input. Must pass in argument subdomainText.", 400) + + # First validate for valid name + subdomain_candidate = SubdomainCandidateName(subdomain_text) + + try: + subdomain_candidate.validate_subdomain() + except ValidationError as e: + return Response({"isValid": False, "message": e.message}) + + # Only check if the subdomain is available if the name is a valid subdomain name + msg = None + + try: + is_valid = subdomain_candidate.is_available() + if not is_valid: + msg = "The subdomain is not available" + except Exception as e: + logger.warn(f"Unable to validate subdomain {subdomain_text}. Error={str(e)}") + is_valid = False + msg = "The subdomain is not available. There was an error during checking availability of the subdomain." + + return Response({"isValid": is_valid, "message": msg}) + + +@api_view(["GET"]) +@permission_classes( + ( + # IsAuthenticated, + ) +) +def get_subdomain_is_available(request: HttpRequest) -> HttpResponse: + """ + Implementation of the API method at endpoint /api/app-subdomain/is-available/ + Supports the GET verb. + + The service contract for the GET action is as follows: + :param str subdomainText: The subdomain text to check for availability. + :returns: An http status code and dict containing {"isAvailable": bool} + + Example request: /api/app-subdomain/is-available/?subdomainText=my-subdomain + """ + subdomain_text = request.GET.get("subdomainText") + + if subdomain_text is None: + return Response("Invalid input. Must pass in argument subdomainText.", 400) + + is_available = False + + try: + subdomain_candidate = SubdomainCandidateName(subdomain_text) + is_available = subdomain_candidate.is_available() + except Exception as e: + logger.warn(f"Unable to validate subdomain {subdomain_text}. Error={str(e)}") + is_available = False + + return Response({"isAvailable": is_available}) + + @api_view(["GET", "POST"]) @permission_classes( ( @@ -842,13 +813,13 @@ def create(self, request, *args, **kwargs): AdminPermission, ) ) -def update_app_status(request): +def update_app_status(request: HttpRequest) -> HttpResponse: """ Manages the app instance status. Implemented as a DRF function based view. Supports GET and POST verbs. - The service contract for the POST actions is as follows: + The service contract for the POST verb is as follows: :param release str: The release id of the app instance, stored in the AppInstance.parameters dict. :param new-status str: The new status code. :param event-ts timestamp: A JSON-formatted timestamp, e.g. 2024-01-25T16:02:50.00Z. diff --git a/apps/.github/dependabot.yml b/apps/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/apps/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/apps/.github/workflows/branch-name-check.yaml b/apps/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 03c9b39bf..000000000 --- a/apps/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v2 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/apps/.github/workflows/code-checks.yaml b/apps/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/apps/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/apps/.gitignore b/apps/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/apps/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/apps/LICENSE b/apps/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/apps/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/apps/MANIFEST.in b/apps/MANIFEST.in deleted file mode 100644 index 1add7d9b5..000000000 --- a/apps/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include apps/templates * diff --git a/apps/README.md b/apps/README.md deleted file mode 100644 index 5cb001d0c..000000000 --- a/apps/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-apps - -Apps Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "apps" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/apps/admin.py b/apps/admin.py index cac291219..a192069fd 100644 --- a/apps/admin.py +++ b/apps/admin.py @@ -1,13 +1,41 @@ +import time + from django.contrib import admin, messages +from django.db.models.query import QuerySet from studio.utils import get_logger -from .models import AppCategories, AppInstance, Apps, AppStatus, ResourceData -from .tasks import deploy_resource +from .helpers import get_URI +from .models import ( + AppCategories, + Apps, + AppStatus, + BaseAppInstance, + CustomAppInstance, + DashInstance, + FilemanagerInstance, + JupyterInstance, + NetpolicyInstance, + RStudioInstance, + ShinyInstance, + Subdomain, + TissuumapsInstance, + VolumeInstance, + VSCodeInstance, +) +from .tasks import delete_resource, deploy_resource logger = get_logger(__name__) +class AppStatusAdmin(admin.ModelAdmin): + list_display = ( + "status", + "time", + ) + list_filter = ["status", "time"] + + class AppsAdmin(admin.ModelAdmin): list_display = ( "name", @@ -22,87 +50,185 @@ class AppsAdmin(admin.ModelAdmin): admin.site.register(Apps, AppsAdmin) -class AppInstanceAdmin(admin.ModelAdmin): - list_display = ("name", "display_owner", "display_project", "state", "access", "app", "display_chart") +class BaseAppAdmin(admin.ModelAdmin): + list_display = ("name", "display_owner", "display_project", "display_status", "display_subdomain", "chart") + + list_filter = ["owner", "project", "app_status__status", "chart"] + actions = ["redeploy_apps", "deploy_resources", "delete_resources"] + + def display_status(self, obj): + status_object = obj.app_status + if status_object: + return status_object.status + else: + "No status" - list_filter = ["owner", "project", "state"] - actions = ["redeploy_apps", "update_chart"] + display_status.short_description = "Status" + + def display_subdomain(self, obj): + subdomain_object = obj.subdomain + if subdomain_object: + return subdomain_object.subdomain + else: + "No Subdomain" + + display_subdomain.short_description = "Subdomain" def display_owner(self, obj): return obj.owner.username + display_owner.short_description = "Owner" + def display_project(self, obj): return obj.project.name - def display_chart(self, obj): - return obj.parameters.get("chart", "No chart") + display_project.short_description = "Project" - @admin.action(description="Redeploy apps") - def redeploy_apps(self, request, queryset): + def display_volumes(self, obj): + if obj.volume is None: + return "No Volumes" + elif isinstance(obj.volume, QuerySet): + return [volume.name for volume in obj.volume.all()] + else: + return obj.volume.name + + display_volumes.short_description = "Volumes" + + @admin.action(description="(Re)deploy resources") + def deploy_resources(self, request, queryset): success_count = 0 failure_count = 0 - for appinstance in queryset: - result = deploy_resource(appinstance.pk) - if result.returncode == 0: - success_count += 1 + for instance in queryset: + instance.set_k8s_values() + instance.url = get_URI(instance.k8s_values) + instance.save(update_fields=["k8s_values", "url"]) + + deploy_resource.delay(instance.serialize()) + time.sleep(2) + info_dict = instance.info + if info_dict: + success = info_dict["helm"].get("success", False) + if success: + success_count += 1 + else: + failure_count += 1 else: failure_count += 1 if success_count: - self.message_user(request, f"{success_count} apps successfully redeployed.", messages.SUCCESS) + self.message_user(request, f"{success_count} apps successfully (re)deployed.", messages.SUCCESS) if failure_count: self.message_user( request, f"Failed to redeploy {failure_count} apps. Check logs for details.", messages.ERROR ) - @admin.action(description="Update helm chart definition in parameters") - def update_chart(self, request, queryset): + @admin.action(description="Delete resources") + def delete_resources(self, request, queryset): success_count = 0 failure_count = 0 - for appinstance in queryset: - # First, update charts for the app - try: - parameters = appinstance.parameters - app = Apps.objects.get(slug=parameters["app_slug"]) - parameters.update({"chart": app.chart}) - - # Secondly, update charts for the dependencies - app_deps = parameters.get("apps") - # Loop through the outer dictionary - for app_key, app_dict in app_deps.items(): - # Loop through each project in the projects dictionary - for key, details in app_dict.items(): - slug = details["slug"] - app = Apps.objects.get(slug=slug) - # Update the chart value - details["chart"] = app.chart - app_deps[app_key][key] = details - parameters.update({"apps": app_deps}) - appinstance.save(update_fields=["parameters"]) - success_count += 1 - except Exception as e: - logger.error(f"Failed to update app {appinstance.name}. Error: {e}") + + for instance in queryset: + instance.set_k8s_values() + delete_resource.delay(instance.serialize()) + info_dict = instance.info + if info_dict: + success = info_dict["helm"].get("success", False) + if success: + success_count += 1 + else: + failure_count += 1 + else: failure_count += 1 + if success_count: - self.message_user(request, f"{success_count} apps successfully updated.", messages.SUCCESS) + self.message_user(request, f"{success_count} apps successfully deleted.", messages.SUCCESS) if failure_count: self.message_user( - request, f"Failed to update {failure_count} apps. Check logs for details.", messages.ERROR + request, f"Failed to delete {failure_count} apps. Check logs for details.", messages.ERROR ) -class AppStatusAdmin(admin.ModelAdmin): - list_display = ( - "appinstance", - "status_type", - "time", +@admin.register(BaseAppInstance) +class BaseAppInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("display_subclass",) + + def display_subclass(self, obj): + subclasses = BaseAppInstance.__subclasses__() + for subclass in subclasses: + app_type = getattr(obj, subclass.__name__.lower(), None) + if app_type: + return app_type.__class__.__name__ + + display_subclass.short_description = "Subclass" + + +@admin.register(RStudioInstance) +class RStudioInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("access", "display_volumes") + + +@admin.register(VSCodeInstance) +class VSCodeInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("access", "display_volumes") + + +@admin.register(JupyterInstance) +class JupyterInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("access", "display_volumes") + + +@admin.register(VolumeInstance) +class VolumeInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("display_size",) + + def display_size(self, obj): + return f"{str(obj.size)} GB" + + display_size.short_description = "Size" + + +@admin.register(NetpolicyInstance) +class NetpolicyInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + + +@admin.register(DashInstance) +class DashInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("image",) + + +@admin.register(CustomAppInstance) +class CustomAppInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ( + "display_volumes", + "image", + "port", + "user_id", ) - list_filter = ["appinstance", "status_type", "time"] + +@admin.register(ShinyInstance) +class ShinyInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ( + "image", + "port", + ) + + +@admin.register(TissuumapsInstance) +class TissuumapsInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ("display_volumes",) + + +@admin.register(FilemanagerInstance) +class FilemanagerInstanceAdmin(BaseAppAdmin): + list_display = BaseAppAdmin.list_display + ( + "display_volumes", + "persistent", + ) -admin.site.register(AppInstance, AppInstanceAdmin) +admin.site.register(Subdomain) admin.site.register(AppCategories) -admin.site.register(ResourceData) admin.site.register(AppStatus, AppStatusAdmin) diff --git a/apps/app_registry.py b/apps/app_registry.py new file mode 100644 index 000000000..4721fdfd4 --- /dev/null +++ b/apps/app_registry.py @@ -0,0 +1,39 @@ +from apps.forms import ( + CustomAppForm, + DashForm, + FilemanagerForm, + JupyterForm, + NetpolicyForm, + RStudioForm, + ShinyForm, + TissuumapsForm, + VolumeForm, + VSCodeForm, +) +from apps.models import ( + CustomAppInstance, + DashInstance, + FilemanagerInstance, + JupyterInstance, + NetpolicyInstance, + RStudioInstance, + ShinyInstance, + TissuumapsInstance, + VolumeInstance, + VSCodeInstance, +) +from apps.types_.app_registry import AppRegistry +from apps.types_.app_types import ModelFormTuple + +APP_REGISTRY = AppRegistry() +APP_REGISTRY.register("jupyter-lab", ModelFormTuple(JupyterInstance, JupyterForm)) +APP_REGISTRY.register("rstudio", ModelFormTuple(RStudioInstance, RStudioForm)) +APP_REGISTRY.register("vscode", ModelFormTuple(VSCodeInstance, VSCodeForm)) +APP_REGISTRY.register("volumeK8s", ModelFormTuple(VolumeInstance, VolumeForm)) +APP_REGISTRY.register("netpolicy", ModelFormTuple(NetpolicyInstance, NetpolicyForm)) +APP_REGISTRY.register("dashapp", ModelFormTuple(DashInstance, DashForm)) +APP_REGISTRY.register("customapp", ModelFormTuple(CustomAppInstance, CustomAppForm)) +APP_REGISTRY.register("shinyapp", ModelFormTuple(ShinyInstance, ShinyForm)) +APP_REGISTRY.register("shinyproxyapp", ModelFormTuple(ShinyInstance, ShinyForm)) +APP_REGISTRY.register("tissuumaps", ModelFormTuple(TissuumapsInstance, TissuumapsForm)) +APP_REGISTRY.register("filemanager", ModelFormTuple(FilemanagerInstance, FilemanagerForm)) diff --git a/apps/apps.py b/apps/apps.py index 84fe2c8dc..ec178376b 100644 --- a/apps/apps.py +++ b/apps/apps.py @@ -4,3 +4,6 @@ class AppsConfig(AppConfig): default_auto_field = "django.db.models.BigAutoField" name = "apps" + + def ready(self): + import apps.signals diff --git a/apps/constants.py b/apps/constants.py new file mode 100644 index 000000000..08dec54f7 --- /dev/null +++ b/apps/constants.py @@ -0,0 +1,21 @@ +HELP_MESSAGE_MAP = { + "name": "Display name for the application. This is the name visible on the app catalogue if the app is public", + "description": "Summarize the application in a few words", + "subdomain": "Valid subdomain names have minimum length of 3 characters and may contain lower case letters a-z \ + and numbers 0-9 and a hyphen '-'. The hyphen should not be at the start or end of the subdomain.", + "access": "Public apps will be displayed on the app catalogue and can be accessed by anyone that has the link to \ + them. Project apps can only be accessed by project members. Private apps are only accessible by users that \ + create the apps.", + "source_code_url": "Provide a URL where the full source code of your app can be accessed (for example, to a GitHub \ + repository or a Figshare or Zenodo entry).", + "flavor": "Hardware allocation for your app. Only one option is available by default. If your app requires more \ + hardware resources, get in touch with us (serve@scilifelab.se) with a request.", + "image": "Docker Image for the app uploaded to DockerHub or GitHub. Each version of your app should have a unique \ + tag.", + "path": "Specify the path inside the container that you want to be persistent (path to database or similar). If \ + you follow our guide to build the container, then please include the username in the path as well.", + "port": "Port that the docker container exposes and the application runs on. This should be an integer between \ + 3000-9999.", + "note_on_linkonly_privacy": "This option can be used only for a limited amount of time, for example while under \ + development or during peer review.", +} diff --git a/apps/controller.py b/apps/controller.py deleted file mode 100644 index 74ce0ae0e..000000000 --- a/apps/controller.py +++ /dev/null @@ -1,151 +0,0 @@ -import json -import os -import subprocess -import tarfile -import uuid - -import yaml -from django.conf import settings - -from studio.utils import get_logger - -from .models import Apps - -KUBEPATH = settings.KUBECONFIG -logger = get_logger(__name__) - - -def delete(options): - logger.info("DELETE FROM CONTROLLER") - # building args for the equivalent of helm uninstall command - args = ["helm", "-n", options["namespace"], "delete", options["release"]] - result = subprocess.run(args, capture_output=True) - return result - - -def deploy(options): - logger.info("STARTING DEPLOY FROM CONTROLLER") - - if "ghcr" in options["chart"]: - version = options["chart"].split(":")[-1] - chart = "oci://" + options["chart"].split(":")[0] - else: - version = None - chart = "charts/" + options["chart"] - - if "release" not in options: - logger.info("Release option not specified.") - return json.dumps({"status": "failed", "reason": "Option release not set."}) - if "appconfig" in options: - # check if path is root path - if "path" in options["appconfig"]: - if "/" == options["appconfig"]["path"]: - logger.info("Root path cannot be copied.") - return json.dumps({"status": "failed", "reason": "Cannot copy / root path."}) - # check if valid userid - if "userid" in options["appconfig"]: - try: - userid = int(options["appconfig"]["userid"]) - except Exception: - logger.error("Userid not a number.", exc_info=True) - return json.dumps({"status": "failed", "reason": "Userid not an integer."}) - if userid > 1010 or userid < 999: - logger.info("Userid outside of allowed range.") - return json.dumps({"status": "failed", "reason": "Userid outside of allowed range."}) - else: - # if no userid, then add default id of 1000 - options["appconfig"]["userid"] = "1000" - # check if valid port - if "port" in options["appconfig"]: - try: - port = int(options["appconfig"]["port"]) - except Exception: - logger.error("Port not a number.", exc_info=True) - return json.dumps({"status": "failed", "reason": "Port not an integer."}) - if port > 9999 or port < 3000: - logger.info("Port outside of allowed range.") - return json.dumps({"status": "failed", "reason": "Port outside of allowed range."}) - # check if valid proxyheartbeatrate - if "proxyheartbeatrate" in options["appconfig"]: - try: - proxyheartbeatrate = int(options["appconfig"]["proxyheartbeatrate"]) - except Exception: - logger.error("Proxy heartbeat rate not a number.", exc_info=True) - return json.dumps({"status": "failed", "reason": "Proxyheartbeatrate not an integer."}) - if proxyheartbeatrate < 1: - logger.info("Heartbeat rate outside of allowed range, must be at least 1.") - return json.dumps( - {"status": "failed", "reason": "Heartbeat rate outside of allowed range, must be at least 1."} - ) - else: - options["appconfig"]["proxyheartbeatrate"] = "10000" - # check if valid proxyheartbeattimeout - if "proxyheartbeattimeout" in options["appconfig"]: - try: - proxyheartbeattimeout = int(options["appconfig"]["proxyheartbeattimeout"]) - except Exception: - logger.error("Proxy heartbeat timeout not a number.", exc_info=True) - return json.dumps({"status": "failed", "reason": "Proxyheartbeattimeout not an integer."}) - if proxyheartbeattimeout < -1 or proxyheartbeattimeout == 0: - logger.info("Heartbeat timeout outside of allowed range, cannot be lower than 0 except for -1.") - return json.dumps( - { - "status": "failed", - "reason": "Heartbeat timeout outside of allowed range, , cannot be lower than 0 except for -1.", - } - ) - else: - options["appconfig"]["proxyheartbeattimeout"] = "60000" - # check if valid proxycontainerwaittime - if "proxycontainerwaittime" in options["appconfig"]: - try: - proxycontainerwaittime = int(options["appconfig"]["proxycontainerwaittime"]) - except Exception: - logger.error("Proxy container wait time not a number.", exc_info=True) - return json.dumps({"status": "failed", "reason": "Proxycontainerwaittime not an integer."}) - if proxycontainerwaittime < 20000: - logger.info("Proxy container wait time outside of allowed range, must be at least 20000.") - return json.dumps( - { - "status": "failed", - "reason": "Proxycontainerwaittime outside of allowed range, must be at least 20000.", - } - ) - else: - options["appconfig"]["proxycontainerwaittime"] = "30000" - - # Save helm values file for internal reference - unique_filename = "charts/values/{}-{}.yaml".format(str(uuid.uuid4()), str(options["app_name"])) - f = open(unique_filename, "w") - f.write(yaml.dump(options)) - f.close() - - # building args for the equivalent of helm install command - args = [ - "helm", - "upgrade", - "--install", - "-n", - options["namespace"], - options["release"], - chart, - "-f", - unique_filename, - ] - - # Append version if deploying via ghcr - if version: - args.append("--version") - args.append(version) - args.append("--repository-cache"), - args.append("/app/charts/.cache/helm/repository") - - logger.info("CONTROLLER: RUNNING HELM COMMAND... ") - - result = subprocess.run(args, capture_output=True) - - # remove file - rm_args = ["rm", unique_filename] - subprocess.run(rm_args) - - return result diff --git a/apps/forms/__init__.py b/apps/forms/__init__.py new file mode 100644 index 000000000..b388094aa --- /dev/null +++ b/apps/forms/__init__.py @@ -0,0 +1,12 @@ +from .custom_field import CustomField # isort:skip +from .base import AppBaseForm, BaseForm +from .custom import CustomAppForm +from .dash import DashForm +from .filemanager import FilemanagerForm +from .jupyter import JupyterForm +from .netpolicy import NetpolicyForm +from .rstudio import RStudioForm +from .shiny import ShinyForm +from .tissuumaps import TissuumapsForm +from .volumes import VolumeForm +from .vscode import VSCodeForm diff --git a/apps/forms/base.py b/apps/forms/base.py new file mode 100644 index 000000000..8341e69ef --- /dev/null +++ b/apps/forms/base.py @@ -0,0 +1,186 @@ +import uuid + +from crispy_forms.helper import FormHelper +from crispy_forms.layout import Button, Div, Submit +from django import forms +from django.shortcuts import get_object_or_404 + +from apps.constants import HELP_MESSAGE_MAP +from apps.forms import CustomField +from apps.models import BaseAppInstance, Subdomain, VolumeInstance +from apps.types_.subdomain import SubdomainCandidateName +from projects.models import Flavor, Project + +__all__ = ["BaseForm", "AppBaseForm"] + + +class BaseForm(forms.ModelForm): + """The most generic form for apps running on serve. Current intended use is for VolumesK8S type apps""" + + subdomain = forms.CharField( + required=False, + min_length=3, + max_length=53, + widget=forms.TextInput(attrs={"style": "text-transform:lowercase;"}), + ) + + def __init__(self, *args, **kwargs): + self.project_pk = kwargs.pop("project_pk", None) + self.project = get_object_or_404(Project, pk=self.project_pk) if self.project_pk else None + self.model_name = self._meta.model._meta.verbose_name.replace("Instance", "") + + super().__init__(*args, **kwargs) + + self._setup_form_fields() + self._setup_form_helper() + + def _setup_form_fields(self): + # Populate subdomain field with instance subdomain if it exists + if self.instance and self.instance.pk: + self.fields["subdomain"].initial = self.instance.subdomain.subdomain + + # Handle name + self.fields["name"].initial = "" + + def _setup_form_helper(self): + # Create a footer for submit form or cancel + self.footer = Div( + Button("cancel", "Cancel", css_class="btn-danger", onclick="window.history.back()"), + Submit("submit", "Submit"), + css_class="card-footer d-flex justify-content-between", + ) + self.helper = FormHelper(self) + self.helper.form_method = "post" + + def clean_subdomain(self): + cleaned_data = super().clean() + subdomain_input = cleaned_data.get("subdomain") + return self.validate_subdomain(subdomain_input) + + def clean_source_code_url(self): + cleaned_data = super().clean() + access = cleaned_data.get("access") + source_code_url = cleaned_data.get("source_code_url") + + if access == "public" and not source_code_url: + self.add_error("source_code_url", "Source is required when access is public.") + + return source_code_url + + def clean_note_on_linkonly_privacy(self): + cleaned_data = super().clean() + + access = cleaned_data.get("access", None) + note_on_linkonly_privacy = cleaned_data.get("note_on_linkonly_privacy", None) + + if access == "link" and not note_on_linkonly_privacy: + self.add_error( + "note_on_linkonly_privacy", "Please, provide a reason for making the app accessible only via a link." + ) + + return note_on_linkonly_privacy + + def validate_subdomain(self, subdomain_input): + # If user did not input subdomain, set it to our standard release name + if not subdomain_input: + subdomain = "r" + uuid.uuid4().hex[0:8] + if Subdomain.objects.filter(subdomain=subdomain_input).exists(): + error_message = "Wow, you just won the lottery. Contact us for a free chocolate bar." + raise forms.ValidationError(error_message) + return subdomain + + # Check if the instance has an existing subdomain + current_subdomain = getattr(self.instance, "subdomain", None) + + # Validate if the subdomain input matches the instance's current subdomain + if current_subdomain and current_subdomain.subdomain == subdomain_input: + return subdomain_input + + # Convert the subdomain to lowercase. OK because we force convert to lowecase in the UI. + subdomain_input = subdomain_input.lower() + + # Check if the subdomain adheres to helm rules + subdomain_candidate = SubdomainCandidateName(subdomain_input) + + try: + subdomain_candidate.validate_subdomain() + except forms.ValidationError as e: + raise forms.ValidationError(f"{e.message}") + + # Check for subdomain availability + if not subdomain_candidate.is_available(): + error_message = "Subdomain already exists. Please choose another one." + raise forms.ValidationError(error_message) + + return subdomain_input + + def get_common_field(self, field_name: str, **kwargs): + """ + This function is very useful because it allows you to create a custom field, + that has a question_mark with tooltip next to the label. So "Name (?)" will have a tooltip. + The text in the tooltip is defined in HELP_MESSAGE_MAP. + The CustomField class just inherits the crispy_forms.layout.Field class and adds the + help_message attribute to it. The template then uses it to render the tooltip for all fields + using this class. + """ + + spinner = kwargs.pop("spinner", False) + + template = "apps/custom_field.html" + base_args = dict( + css_class="form-control form-control-with-spinner" if spinner else "form-control", + wrapper_class="mb-3", + rows=3, + help_message=HELP_MESSAGE_MAP.get(field_name, ""), + spinner=spinner, + ) + + base_args.update(kwargs) + field = CustomField(field_name, **base_args) + field.set_template(template) + return Div(field, css_class="form-input-with-spinner" if spinner else None) + + class Meta: + # Specify model to be used + model = BaseAppInstance + fields = "__all__" + + +class AppBaseForm(BaseForm): + """ + Generic form for apps that require some compute power, + so you can treat this form as an actual base form for the most of the apps + """ + + volume = forms.ModelChoiceField( + queryset=VolumeInstance.objects.none(), required=False, empty_label="None", initial=None + ) + + flavor = forms.ModelChoiceField(queryset=Flavor.objects.none(), required=True, empty_label=None) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + def _setup_form_fields(self): + super()._setup_form_fields() + flavor_queryset = ( + Flavor.objects.filter(project__pk=self.project_pk) if self.project_pk else Flavor.objects.none() + ) + # Handle Flavor field + self.fields["flavor"].label = "Hardware" + self.fields["flavor"].queryset = flavor_queryset + self.fields["flavor"].initial = flavor_queryset.first() # if flavor_queryset else None + + # Handle Access field + self.fields["access"].label = "Permission" + + # Handle Volume field + volume_queryset = ( + VolumeInstance.objects.filter(project__pk=self.project_pk) + if self.project_pk + else VolumeInstance.objects.none() + ) + + self.fields["volume"].queryset = volume_queryset + self.fields["volume"].initial = volume_queryset + self.fields["volume"].help_text = f"Select a volume to attach to your {self.model_name}." diff --git a/apps/forms/custom.py b/apps/forms/custom.py new file mode 100644 index 000000000..4fd616502 --- /dev/null +++ b/apps/forms/custom.py @@ -0,0 +1,86 @@ +from crispy_forms.layout import HTML, Div, Field, Layout, MultiField +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import CustomAppInstance, VolumeInstance +from projects.models import Flavor + +__all__ = ["CustomAppForm"] +from apps.forms import CustomField + + +class CustomAppForm(AppBaseForm): + flavor = forms.ModelChoiceField(queryset=Flavor.objects.none(), required=False, empty_label=None) + port = forms.IntegerField(min_value=3000, max_value=9999, required=True) + image = forms.CharField(max_length=255, required=True) + path = forms.CharField(max_length=255, required=False) + + def _setup_form_fields(self): + # Handle Volume field + super()._setup_form_fields() + self.fields["volume"].initial = None + + def _setup_form_helper(self): + super()._setup_form_helper() + + body = Div( + self.get_common_field("name", placeholder="Name your app"), + self.get_common_field("description", rows=3), + self.get_common_field( + "subdomain", placeholder="Enter a subdomain or leave blank for a random one", spinner=True + ), + self.get_common_field("volume"), + self.get_common_field("path", placeholder="/home/..."), + self.get_common_field("flavor"), + self.get_common_field("access"), + self.get_common_field("source_code_url", placeholder="Provide a link to the public source code"), + self.get_common_field( + "note_on_linkonly_privacy", + rows=1, + placeholder="Describe why you want to make the app accessible only via a link", + ), + self.get_common_field("port", placeholder="8000"), + self.get_common_field("image"), + Field("tags"), + css_class="card-body", + ) + self.helper.layout = Layout(body, self.footer) + + def clean_path(self): + cleaned_data = super().clean() + + path = cleaned_data.get("path", None) + volume = cleaned_data.get("volume", None) + + if volume and not path: + self.add_error("path", "Path is required when volume is selected.") + + if path and not volume: + self.add_error("path", "Warning, you have provided a path, but not selected a volume.") + + if path: + # If new path matches current path, it is valid. + if self.instance and getattr(self.instance, "path", None) == path: + return path + # Verify that path starts with "/home" + path = path.strip().rstrip("/").lower().replace(" ", "") + if not path.startswith("/home"): + self.add_error("path", 'Path must start with "/home"') + + return path + + class Meta: + model = CustomAppInstance + fields = [ + "name", + "description", + "volume", + "path", + "flavor", + "access", + "note_on_linkonly_privacy", + "source_code_url", + "port", + "image", + "tags", + ] diff --git a/apps/forms/custom_field.py b/apps/forms/custom_field.py new file mode 100644 index 000000000..59a599273 --- /dev/null +++ b/apps/forms/custom_field.py @@ -0,0 +1,17 @@ +from crispy_forms.layout import Field + + +class CustomField(Field): + template = "apps/custom_field.html" + + def __init__(self, *args, **kwargs): + self.help_message = kwargs.pop("help_message", "") + self.spinner = kwargs.pop("spinner", False) + super().__init__(*args, **kwargs) + + def render(self, form, context, **kwargs): + context.update({"help_message": self.help_message, "spinner": self.spinner}) + return super().render(form, context, **kwargs) + + def set_template(self, name: str): + self.template = name diff --git a/apps/forms/dash.py b/apps/forms/dash.py new file mode 100644 index 000000000..bfead452b --- /dev/null +++ b/apps/forms/dash.py @@ -0,0 +1,65 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import DashInstance +from projects.models import Flavor + +__all__ = ["DashForm"] + + +class DashForm(AppBaseForm): + flavor = forms.ModelChoiceField(queryset=Flavor.objects.none(), required=False, empty_label=None) + port = forms.IntegerField(min_value=3000, max_value=9999, required=True) + image = forms.CharField(max_length=255, required=True) + + def _setup_form_fields(self): + # Handle Volume field + super()._setup_form_fields() + + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + self.get_common_field("name", placeholder="Name your app"), + self.get_common_field("description", rows="3", placeholder="Provide a detailed description of your app"), + self.get_common_field( + "subdomain", placeholder="Enter a subdomain or leave blank for a random one", spinner=True + ), + self.get_common_field("flavor"), + self.get_common_field("access"), + self.get_common_field( + "note_on_linkonly_privacy", + placeholder="Describe why you want to make the app accessible only via a link", + ), + self.get_common_field("source_code_url", placeholder="Provide a link to the public source code"), + self.get_common_field("port", placeholder="8000"), + self.get_common_field("image", placeholder="registry/repository/image:tag"), + Field("tags"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + def clean(self): + cleaned_data = super().clean() + access = cleaned_data.get("access") + source_code_url = cleaned_data.get("source_code_url") + + if access == "public" and not source_code_url: + self.add_error("source_code_url", "Source is required when access is public.") + + return cleaned_data + + class Meta: + model = DashInstance + fields = [ + "name", + "description", + "flavor", + "access", + "note_on_linkonly_privacy", + "source_code_url", + "port", + "image", + "tags", + ] diff --git a/apps/forms/filemanager.py b/apps/forms/filemanager.py new file mode 100644 index 000000000..7f9676879 --- /dev/null +++ b/apps/forms/filemanager.py @@ -0,0 +1,46 @@ +from crispy_forms.layout import HTML, Button, Div, Field, Layout, Submit +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import FilemanagerInstance, VolumeInstance + +__all__ = ["FilemanagerForm"] + + +class FilemanagerForm(AppBaseForm): + volume = forms.ModelMultipleChoiceField(queryset=VolumeInstance.objects.none(), required=False) + + def _setup_form_helper(self): + super()._setup_form_helper() + + self.footer = Div( + Button("cancel", "Cancel", css_class="btn-danger", onclick="window.history.back()"), + Submit("submit", "Activate", css_class="btn-profile text-dark"), + css_class="card-footer d-flex justify-content-between", + ) + body = Div( + Div( + HTML( + """

You are about to activate file manager on SciLifeLab Serve. + You can use it to upload or download files to a volume associated with this project. + This service will be active for 24 hours and automatically terminated afterwards. + The uploaded files will stay on the volume even after this service has been terminated.

+ """ + ), + HTML("

Click 'Activate' to activate file manager

"), + css_class="p-3 my-3", + ), + Field("name", type="hidden"), + Field("access", type="hidden"), + Field("flavor", type="hidden"), + Field("volume"), + css_class="card-body", + ) + + self.fields["name"].initial = "File Manager" + self.fields["access"].initial = "project" + self.helper.layout = Layout(body, self.footer) + + class Meta: + model = FilemanagerInstance + fields = ["name", "access", "flavor", "volume"] diff --git a/apps/forms/jupyter.py b/apps/forms/jupyter.py new file mode 100644 index 000000000..fea95a339 --- /dev/null +++ b/apps/forms/jupyter.py @@ -0,0 +1,29 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import JupyterInstance, VolumeInstance +from projects.models import Flavor + +__all__ = ["JupyterForm"] + + +class JupyterForm(AppBaseForm): + volume = forms.ModelMultipleChoiceField(queryset=VolumeInstance.objects.none(), required=False) + + def _setup_form_helper(self): + super()._setup_form_helper() + + body = Div( + self.get_common_field("name", placeholder="Name your app"), + Field("volume"), + self.get_common_field("access"), + self.get_common_field("flavor"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + class Meta: + model = JupyterInstance + fields = ["name", "volume", "flavor", "access"] diff --git a/apps/forms/netpolicy.py b/apps/forms/netpolicy.py new file mode 100644 index 000000000..821a72ae8 --- /dev/null +++ b/apps/forms/netpolicy.py @@ -0,0 +1,19 @@ +from crispy_forms.layout import Div, Field, Layout + +from apps.forms.base import BaseForm +from apps.models import NetpolicyInstance + +__all__ = ["NetpolicyForm"] + + +class NetpolicyForm(BaseForm): + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div(Field("name", placeholder="Name your app"), css_class="card-body") + + self.helper.layout = Layout(body, self.footer) + + # create meta class + class Meta: + model = NetpolicyInstance + fields = ["name"] diff --git a/apps/forms/rstudio.py b/apps/forms/rstudio.py new file mode 100644 index 000000000..39317c62c --- /dev/null +++ b/apps/forms/rstudio.py @@ -0,0 +1,27 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import RStudioInstance, VolumeInstance + +__all__ = ["RStudioForm"] + + +class RStudioForm(AppBaseForm): + volume = forms.ModelMultipleChoiceField(queryset=VolumeInstance.objects.none(), required=False) + + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + self.get_common_field("name", placeholder="Name your app"), + Field("volume"), + self.get_common_field("flavor"), + self.get_common_field("access"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + class Meta: + model = RStudioInstance + fields = ["name", "volume", "flavor", "access"] diff --git a/apps/forms/shiny.py b/apps/forms/shiny.py new file mode 100644 index 000000000..74fc9991c --- /dev/null +++ b/apps/forms/shiny.py @@ -0,0 +1,95 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import ShinyInstance +from projects.models import Flavor + +__all__ = ["ShinyForm"] + + +class ShinyForm(AppBaseForm): + flavor = forms.ModelChoiceField(queryset=Flavor.objects.none(), required=False, empty_label=None) + port = forms.IntegerField(min_value=3000, max_value=9999, required=True) + image = forms.CharField(max_length=255, required=True) + proxy = forms.BooleanField( + required=False, + initial=False, + label="Activate Proxy", + help_text=""" Check this box if your + app requires lots of compute power. This will enable a proxy to handle the load. + """, + ) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + if self.instance and self.instance.pk: + self.initial_subdomain = self.instance.subdomain.subdomain + self.initial_proxy = self.instance.proxy + + def _setup_form_fields(self): + # Handle Volume field + super()._setup_form_fields() + + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + self.get_common_field("name", placeholder="Name your app"), + self.get_common_field("description", rows="3", placeholder="Provide a detailed description of your app"), + Field( + "proxy", + ), + self.get_common_field( + "subdomain", placeholder="Enter a subdomain or leave blank for a random one", spinner=True + ), + self.get_common_field("flavor"), + self.get_common_field("access"), + self.get_common_field( + "note_on_linkonly_privacy", + placeholder="Describe why you want to make the app accessible only via a link", + ), + self.get_common_field("source_code_url", placeholder="Provide a link to the public source code"), + self.get_common_field("port", placeholder="3838"), + self.get_common_field("image", placeholder="registry/repository/image:tag"), + Field("tags"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + def clean(self): + cleaned_data = super().clean() + access = cleaned_data.get("access", None) + source_code_url = cleaned_data.get("source_code_url", None) + + if access == "public" and not source_code_url: + self.add_error("source_code_url", "Source is required when access is public.") + + proxy = cleaned_data.get("proxy") + subdomain = cleaned_data.get("subdomain") + + # Check if boolfield has changed + if self.instance and self.instance.pk: + if proxy != self.initial_proxy: + # Ensure charfield is also changed + if subdomain == self.initial_subdomain: + self.add_error("subdomain", "Subdomain must be changed if proxy is changed.") + + return cleaned_data + + class Meta: + model = ShinyInstance + fields = [ + "name", + "description", + "proxy", + "volume", + "flavor", + "access", + "note_on_linkonly_privacy", + "source_code_url", + "port", + "image", + "tags", + ] diff --git a/apps/forms/tissuumaps.py b/apps/forms/tissuumaps.py new file mode 100644 index 000000000..5321cf6b9 --- /dev/null +++ b/apps/forms/tissuumaps.py @@ -0,0 +1,36 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import TissuumapsInstance +from projects.models import Flavor + +__all__ = ["TissuumapsForm"] + + +class TissuumapsForm(AppBaseForm): + def _setup_form_fields(self): + # Handle Volume field + super()._setup_form_fields() + self.fields["volume"].initial = None + + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + self.get_common_field("name", placeholder="Name your app"), + self.get_common_field("description", rows="3", placeholder="Provide a detailed description of your app"), + self.get_common_field( + "subdomain", placeholder="Enter a subdomain or leave blank for a random one", spinner=True + ), + Field("volume"), + self.get_common_field("flavor"), + self.get_common_field("access"), + Field("tags"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + class Meta: + model = TissuumapsInstance + fields = ["name", "description", "volume", "flavor", "access", "tags"] diff --git a/apps/forms/volumes.py b/apps/forms/volumes.py new file mode 100644 index 000000000..fdfd83040 --- /dev/null +++ b/apps/forms/volumes.py @@ -0,0 +1,24 @@ +from crispy_forms.layout import Div, Field, Layout + +from apps.forms.base import BaseForm +from apps.models import VolumeInstance + +__all__ = ["VolumeForm"] + + +class VolumeForm(BaseForm): + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + Field("name", placeholder="Name your app"), + Field("size"), + Field("subdomain", placeholder="Enter a subdomain or leave blank for a random one"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + # create meta class + class Meta: + model = VolumeInstance + fields = ["name", "size"] diff --git a/apps/forms/vscode.py b/apps/forms/vscode.py new file mode 100644 index 000000000..8d1f9745e --- /dev/null +++ b/apps/forms/vscode.py @@ -0,0 +1,27 @@ +from crispy_forms.layout import HTML, Div, Field, Layout +from django import forms + +from apps.forms.base import AppBaseForm +from apps.models import VolumeInstance, VSCodeInstance + +__all__ = ["VSCodeForm"] + + +class VSCodeForm(AppBaseForm): + volume = forms.ModelMultipleChoiceField(queryset=VolumeInstance.objects.none(), required=False) + + def _setup_form_helper(self): + super()._setup_form_helper() + body = Div( + self.get_common_field("name", placeholder="Name your app"), + Field("volume"), + self.get_common_field("flavor"), + self.get_common_field("access"), + css_class="card-body", + ) + + self.helper.layout = Layout(body, self.footer) + + class Meta: + model = VSCodeInstance + fields = ["name", "volume", "flavor", "access"] diff --git a/apps/generate_form.py b/apps/generate_form.py deleted file mode 100644 index 9e6bd54a1..000000000 --- a/apps/generate_form.py +++ /dev/null @@ -1,280 +0,0 @@ -from django.conf import settings -from django.db.models import Case, IntegerField, Q, Value, When - -from models.models import Model -from projects.models import S3, Environment, Flavor, ReleaseName -from studio.utils import get_logger - -from .models import AppInstance, Apps - -logger = get_logger(__name__) - -key_words = [ - "appobj", - "model", - "flavor", - "environment", - "volumes", - "apps", - "logs", - "permissions", - "default_values", - "export-cli", - "csrfmiddlewaretoken", - "S3", - "env_variables", - "publishable", -] - - -def get_form_models(aset, project, appinstance=[]): - dep_model = False - models = [] - if "model" in aset: - logger.info("app requires a model") - dep_model = True - if "object_type" in aset["model"]: - object_type = aset["model"]["object_type"] - else: - object_type = "default" - models = Model.objects.filter(project=project, object_type__slug=object_type) - - for model in models: - if appinstance and model.appinstance_set.filter(pk=appinstance.pk).exists(): - logger.info(model) - model.selected = "selected" - else: - model.selected = "" - return dep_model, models - - -def get_form_apps(aset, project, myapp, user, appinstance=[]): - dep_apps = False - app_deps = [] - if "apps" in aset: - dep_apps = True - app_deps = dict() - apps = aset["apps"] - for app_name, option_type in apps.items(): - logger.info(">>>>>") - logger.info(app_name) - # .order_by('-revision').first() - app_obj = Apps.objects.filter(name=app_name) - logger.info(app_obj) - logger.info(">>>>>") - # TODO: Only get app instances that we have permission to list. - - app_instances = AppInstance.objects.get_available_app_dependencies( - user=user, project=project, app_name=app_name - ) - # TODO: Special case here for "environment" app. - # Could be solved by supporting "condition": - # '"appobj.app_slug":"true"' - if app_name == "Environment": - app_instances = AppInstance.objects.filter( - ~Q(state="Deleted"), - Q(owner=user) | Q(access__in=["project", "public"]), - project=project, - app__name=app_name, - parameters__contains={"appobj": {myapp.slug: True}}, - ) - - for ain in app_instances: - if appinstance and ain.appinstance_set.filter(pk=appinstance.pk).exists(): - ain.selected = "selected" - else: - ain.selected = "" - - if option_type == "one": - app_deps[app_name] = { - "instances": app_instances, - "option_type": "", - } - else: - app_deps[app_name] = { - "instances": app_instances, - "option_type": "multiple", - } - return dep_apps, app_deps - - -def get_disable_fields(): - try: - result = settings.DISABLED_APP_INSTANCE_FIELDS - return result if result is not None else [] - except Exception: - return [] - - -def get_form_primitives(app_settings, appinstance=[]): - disabled_fields = get_disable_fields() - - all_keys = app_settings.keys() - primitives = dict() - - for key in all_keys: - if key not in key_words: - primitives[key] = app_settings[key] - if "meta" in primitives[key]: - primitives[key]["meta_title"] = primitives[key]["meta"]["title"] - else: - primitives[key]["meta_title"] = key - - for disabled_field in disabled_fields: - if disabled_field in primitives[key]: - del primitives[key][disabled_field] - - if appinstance and key in appinstance.parameters.keys(): - for _key, _ in app_settings[key].items(): - is_meta_key = _key in ["meta", "meta_title"] - if not is_meta_key: - parameters_of_key = appinstance.parameters[key] - - logger.info("_key: %s", _key) - - if _key in parameters_of_key.keys(): - primitives[key][_key]["default"] = parameters_of_key[_key] - - return primitives - - -def get_form_permission(aset, project, appinstance=[]): - form_permissions = { - "project": {"value": "false", "option": "false"}, - "public": {"value": "false", "option": "false"}, - "link": {"value": "false", "option": "false"}, - "private": {"value": "true", "option": "true"}, - } - dep_permissions = True - if "permissions" in aset: - form_permissions["project"] = aset["permissions"]["project"] - form_permissions["public"] = aset["permissions"]["public"] - form_permissions["link"] = aset["permissions"]["link"] - form_permissions["private"] = aset["permissions"]["private"] - - # I don't really know why we keep this - # if not form_permissions: - # dep_permissions = False - - if appinstance: - try: - ai_vals = appinstance.parameters - logger.info(ai_vals["permissions"]) - form_permissions["public"]["value"] = ai_vals["permissions"].get("public", False) - form_permissions["project"]["value"] = ai_vals["permissions"].get("project", False) - form_permissions["private"]["value"] = ai_vals["permissions"].get("private", False) - form_permissions["link"]["value"] = ai_vals["permissions"].get("link", False) - logger.info(form_permissions) - except Exception: - logger.error("Permissions not set for app instance, using default.", exc_info=True) - - return dep_permissions, form_permissions - - -# TODO: refactor. Change default value to immutable -def get_form_appobj(aset, project, appinstance=[]): - logger.info("CHECKING APP OBJ") - dep_appobj = False - appobjs = dict() - if "appobj" in aset: - logger.info("NEEDS APP OBJ") - dep_appobj = True - appobjs["objs"] = Apps.objects.all() - appobjs["title"] = aset["appobj"]["title"] - appobjs["type"] = aset["appobj"]["type"] - - logger.info(appobjs) - return dep_appobj, appobjs - - -def get_form_environments(aset, project, app, appinstance=[]): - logger.info("CHECKING ENVIRONMENT") - dep_environment = False - environments = dict() - if "environment" in aset: - dep_environment = True - if aset["environment"]["type"] == "match": - environments["objs"] = Environment.objects.filter( - Q(project=project) | Q(project__isnull=True, public=True), - app__slug=app.slug, - ).order_by( - Case( - When(name__contains="- public", then=Value(1)), - default=Value(0), - output_field=IntegerField(), - ), - "-name", - ) - elif aset["environment"]["type"] == "any": - environments["objs"] = Environment.objects.filter( - Q(project=project) | Q(project__isnull=True, public=True) - ).order_by( - Case( - When(name__contains="- public", then=Value(1)), - default=Value(0), - output_field=IntegerField(), - ), - "-name", - ) - elif "apps" in aset["environment"]: - environments["objs"] = Environment.objects.filter( - Q(project=project) | Q(project__isnull=True, public=True), - app__slug__in=aset["environment"]["apps"], - ).order_by( - Case( - When(name__contains="- public", then=Value(1)), - default=Value(0), - output_field=IntegerField(), - ), - "-name", - ) - - environments["title"] = aset["environment"]["title"] - if appinstance: - ai_vals = appinstance.parameters - environments["selected"] = ai_vals["environment"]["pk"] - - return dep_environment, environments - - -def get_form_S3(aset, project, app, appinstance=[]): - logger.info("CHECKING S3") - dep_S3 = False - s3stores = [] - if "S3" in aset: - dep_S3 = True - s3stores = S3.objects.filter(project=project) - return dep_S3, s3stores - - -def get_link_privacy_type_note(aset, project, appinstance=tuple()): - if appinstance: - return appinstance.note_on_linkonly_privacy - return "" - - -def generate_form(aset, project, app, user, appinstance=[]): - form = dict() - form["dep_model"], form["models"] = get_form_models(aset, project, appinstance) - form["dep_apps"], form["app_deps"] = get_form_apps(aset, project, app, user, appinstance) - form["dep_appobj"], form["appobjs"] = get_form_appobj(aset, project, appinstance) - form["dep_environment"], form["environments"] = get_form_environments(aset, project, app, appinstance) - form["dep_S3"], form["s3stores"] = get_form_S3(aset, project, app, appinstance) - - form["dep_vols"] = False - form["dep_flavor"] = False - if "flavor" in aset: - form["dep_flavor"] = True - if appinstance and appinstance.flavor: - form["current_flavor"] = Flavor.objects.filter(project=project, id=appinstance.flavor.id) - form["flavors"] = Flavor.objects.filter(project=project).exclude(id=appinstance.flavor.id) - else: - form["current_flavor"] = None - form["flavors"] = Flavor.objects.filter(project=project) - - form["primitives"] = get_form_primitives(aset, appinstance) - form["dep_permissions"], form["form_permissions"] = get_form_permission(aset, project, appinstance) - release_names = ReleaseName.objects.filter(project=project, status="active") - form["release_names"] = release_names - form["link_privacy_type_note"] = get_link_privacy_type_note(aset, project, appinstance) - return form diff --git a/apps/helpers.py b/apps/helpers.py index 6c4803b85..9c72ec8c3 100644 --- a/apps/helpers.py +++ b/apps/helpers.py @@ -1,75 +1,22 @@ -import re -import time -import uuid from datetime import datetime from enum import Enum from typing import Optional -from django.apps import apps -from django.conf import settings from django.core.exceptions import ObjectDoesNotExist from django.db import transaction -from django.template import engines -from projects.models import Flavor from studio.utils import get_logger -from .models import AppInstance, AppStatus -from .serialize import serialize_app -from .tasks import deploy_resource +from .models import Apps, AppStatus, BaseAppInstance, Subdomain logger = get_logger(__name__) -ReleaseName = apps.get_model(app_label=settings.RELEASENAME_MODEL) - - -def create_instance_params(instance, action="create"): - logger.info("HELPER - CREATING INSTANCE PARAMS") - RELEASE_NAME = "r" + uuid.uuid4().hex[0:8] - logger.info("RELEASE_NAME: " + RELEASE_NAME) - - SERVICE_NAME = RELEASE_NAME + "-" + instance.app.slug - # TODO: Fix for multicluster setup, look at e.g. labs - HOST = settings.DOMAIN - AUTH_HOST = settings.AUTH_DOMAIN - AUTH_PROTOCOL = settings.AUTH_PROTOCOL - NAMESPACE = settings.NAMESPACE - - # Add some generic parameters. - parameters = { - "release": RELEASE_NAME, - "chart": str(instance.app.chart), - "namespace": NAMESPACE, - "app_slug": str(instance.app.slug), - "app_revision": str(instance.app.revision), - "appname": RELEASE_NAME, - "global": { - "domain": HOST, - "auth_domain": AUTH_HOST, - "protocol": AUTH_PROTOCOL, - }, - "s3sync": {"image": "scaleoutsystems/s3-sync:latest"}, - "service": { - "name": SERVICE_NAME, - "port": instance.parameters["default_values"]["port"], - "targetport": instance.parameters["default_values"]["targetport"], - }, - "storageClass": settings.STORAGECLASS, - } - - instance.parameters.update(parameters) - - if "project" not in instance.parameters: - instance.parameters["project"] = dict() - - instance.parameters["project"].update({"name": instance.project.name, "slug": instance.project.slug}) - - -def can_access_app_instance(app_instance, user, project): + +def can_access_app_instance(instance, user, project): """Checks if a user has access to an app instance Args: - app_instance (AppInstance): app instance object + instance (subclass of BaseAppInstance): instance object user (User): user object project (Project): project object @@ -78,32 +25,32 @@ def can_access_app_instance(app_instance, user, project): """ authorized = False - if app_instance.access in ("public", "link"): + if instance.access in ("public", "link"): authorized = True - elif app_instance.access == "project": + elif instance.access == "project": if user.has_perm("can_view_project", project): authorized = True else: - if user.has_perm("can_access_app", app_instance): + if user.has_perm("can_access_app", instance): authorized = True return authorized -def can_access_app_instances(app_instances, user, project): +def can_access_app_instances(instances, user, project): """Checks if user has access to all app instances provided Args: - app_instances (Queryset): list of app instances + instances (Queryset): list of instances user (User): user object project (Project): project object Returns: Boolean: returns False if user lacks - permission to any of the app instances provided + permission to any of the instances provided """ - for app_instance in app_instances: - authorized = can_access_app_instance(app_instance, user, project) + for instance in instances: + authorized = can_access_app_instance(instance, user, project) if not authorized: return False @@ -135,113 +82,6 @@ def handle_permissions(parameters, project): return access -# TODO: refactor -# 1. data=[]. This is bad as this is not a list, but a dict and secondly, -# it is not a good practice to use mutable as default -# 2. Use some type annotations -# 3. Use tuple as return type instead of list -def create_app_instance(user, project, app, app_settings, data=[], wait=False): - app_name = data.get("app_name") - app_description = data.get("app_description") - created_by_admin = False - # For custom apps, if admin user fills form, then data.get("admin") exists as hidden input - if data.get("created_by_admin"): - created_by_admin = True - parameters_out, app_deps, model_deps = serialize_app(data, project, app_settings, user.username) - parameters_out["created_by_admin"] = created_by_admin - authorized = can_access_app_instances(app_deps, user, project) - - if not authorized: - raise Exception("Not authorized to use specified app dependency") - - access = handle_permissions(parameters_out, project) - - flavor_id = data.get("flavor", None) - flavor = Flavor.objects.get(pk=flavor_id, project=project) if flavor_id else None - - source_code_url = data.get("source_code_url") - app_instance = AppInstance( - name=app_name, - description=app_description, - access=access, - app=app, - project=project, - info={}, - parameters=parameters_out, - owner=user, - flavor=flavor, - note_on_linkonly_privacy=data.get("link_privacy_type_note"), - source_code_url=source_code_url, - ) - - create_instance_params(app_instance, "create") - - # Attempt to create a ReleaseName model object - rel_name_obj = [] - if "app_release_name" in data and data.get("app_release_name") != "": - submitted_rn = data.get("app_release_name") - try: - rel_name_obj = ReleaseName.objects.get(name=submitted_rn, project=project, status="active") - rel_name_obj.status = "in-use" - rel_name_obj.save() - app_instance.parameters["release"] = submitted_rn - except Exception: - logger.error("Submitted release name not owned by project.", exc_info=True) - return [False, None, None] - - # Add fields for apps table: - # to be displayed as app details in views - if app_instance.app.table_field and app_instance.app.table_field != "": - django_engine = engines["django"] - info_field = django_engine.from_string(app_instance.app.table_field).render(app_instance.parameters) - # Nikita Churikov @ 2024-01-25 - # TODO: this seems super bad and exploitable - app_instance.table_field = eval(info_field) - else: - app_instance.table_field = {} - - # Setting status fields before saving app instance - status = AppStatus(appinstance=app_instance) - status.status_type = "Created" - status.info = app_instance.parameters["release"] - if "appconfig" in app_instance.parameters: - if "path" in app_instance.parameters["appconfig"]: - # remove trailing / in all cases - if app_instance.parameters["appconfig"]["path"] != "/": - app_instance.parameters["appconfig"]["path"] = app_instance.parameters["appconfig"]["path"].rstrip("/") - if app_deps: - if not created_by_admin: - app_instance.parameters["appconfig"]["path"] = ( - "/home/" + app_instance.parameters["appconfig"]["path"] - ) - if "userid" not in app_instance.parameters["appconfig"]: - app_instance.parameters["appconfig"]["userid"] = "1000" - if "proxyheartbeatrate" not in app_instance.parameters["appconfig"]: - app_instance.parameters["appconfig"]["proxyheartbeatrate"] = "10000" - if "proxyheartbeattimeout" not in app_instance.parameters["appconfig"]: - app_instance.parameters["appconfig"]["proxyheartbeattimeout"] = "60000" - if "proxycontainerwaittime" not in app_instance.parameters["appconfig"]: - app_instance.parameters["appconfig"]["proxycontainerwaittime"] = "30000" - app_instance.save() - # Saving ReleaseName, status and setting up dependencies - if rel_name_obj: - rel_name_obj.app = app_instance - rel_name_obj.save() - status.save() - app_instance.app_dependencies.set(app_deps) - app_instance.model_dependencies.set(model_deps) - - # Finally, attempting to create apps resources - res = deploy_resource.delay(app_instance.pk, "create") - - # wait is passed as a function parameter - if wait: - while not res.ready(): - time.sleep(0.1) - - return [True, project.slug, app_instance.app.category.slug] - - class HandleUpdateStatusResponseCode(Enum): NO_ACTION = 0 UPDATED_STATUS = 1 @@ -256,7 +96,7 @@ def handle_update_status_request( Helper function to handle update app status requests by determining if the request should be performed or ignored. - :param release str: The release id of the app instance, stored in the AppInstance.parameters dict. + :param release str: The release id of the app instance, stored in the AppInstance.k8s_values dict in the subdomain. :param new_status str: The new status code. Trimmed to max 15 chars if needed. :param event_ts timestamp: A JSON-formatted timestamp in UTC, e.g. 2024-01-25T16:02:50.00Z. :param event_msg json dict: An optional json dict containing pod-msg and/or container-msg. @@ -272,27 +112,27 @@ def handle_update_status_request( # We wrap the select and update tasks in a select_for_update lock # to avoid race conditions. + subdomain = Subdomain.objects.get(subdomain=release) + with transaction.atomic(): - app_instance = ( - AppInstance.objects.select_for_update().filter(parameters__contains={"release": release}).last() - ) - if app_instance is None: + instance = BaseAppInstance.objects.select_for_update().filter(subdomain=subdomain).last() + if instance is None: logger.info("The specified app instance was not found release=%s.", release) raise ObjectDoesNotExist - logger.debug("The app instance exists. name=%s, state=%s", app_instance.name, app_instance.state) + logger.debug("The app instance exists. name=%s", instance.name) # Also get the latest app status object for this app instance - if app_instance.status is None or app_instance.status.count() == 0: + if instance.app_status is None: # Missing app status so create one now logger.info("AppInstance %s does not have an associated AppStatus. Creating one now.", release) - status_object = AppStatus(appinstance=app_instance) - update_status(app_instance, status_object, new_status, event_ts, event_msg) + app_status = AppStatus.objects.create() + update_status(instance, app_status, new_status, event_ts, event_msg) return HandleUpdateStatusResponseCode.CREATED_FIRST_STATUS else: - app_status = app_instance.status.latest() + app_status = instance.app_status - logger.debug("AppStatus %s, %s, %s.", app_status.status_type, app_status.time, app_status.info) + logger.debug("AppStatus %s, %s, %s.", app_status.status, app_status.time, app_status.info) # Now determine whether to update the state and status @@ -308,7 +148,7 @@ def handle_update_status_request( # The event is newer than the existing persisted object - if new_status == app_instance.state: + if new_status == instance.app_status.status: # The same status. Simply update the time. logger.debug("The same status. Simply update the time.") update_status_time(app_status, event_ts, event_msg) @@ -316,8 +156,8 @@ def handle_update_status_request( # Different status and newer time logger.debug("Different status and newer time.") - status_object = AppStatus(appinstance=app_instance) - update_status(app_instance, status_object, new_status, event_ts, event_msg) + status_object = instance.app_status + update_status(instance, status_object, new_status, event_ts, event_msg) return HandleUpdateStatusResponseCode.UPDATED_STATUS except Exception as err: @@ -331,7 +171,7 @@ def update_status(appinstance, status_object, status, status_ts=None, event_msg= Helper function to update the status of an appinstance and a status object. """ # Persist a new app statuss object - status_object.status_type = status + status_object.status = status status_object.time = status_ts status_object.info = event_msg status_object.save() @@ -346,8 +186,8 @@ def update_status(appinstance, status_object, status, status_ts=None, event_msg= status_object.save(update_fields=["time", "info"]) # Update the app instance object - appinstance.state = status - appinstance.save(update_fields=["state"]) + appinstance.app_status = status_object + appinstance.save(update_fields=["app_status"]) @transaction.atomic @@ -362,3 +202,111 @@ def update_status_time(status_object, status_ts, event_msg=None): else: status_object.info = event_msg status_object.save(update_fields=["time", "info"]) + + +def get_URI(values): + URI = "https://" + values["subdomain"] + "." + values["global"]["domain"] + + URI = URI.strip("/") + return URI + + +@transaction.atomic +def create_instance_from_form(form, project, app_slug, app_id=None): + """ + Create or update an instance from a form. This function handles both the creation of new instances + and the updating of existing ones based on the presence of an app_id. + + Parameters: + - form: The form instance containing validated data. + - project: The project to which this instance belongs. + - app_slug: Slug of the app associated with this instance. + - app_id: Optional ID of an existing instance to update. If None, a new instance is created. + + Returns: + - The newly created or updated instance. + + Raises: + - ValueError: If the form does not have a 'subdomain' or if the specified app cannot be found. + """ + from .tasks import deploy_resource + + subdomain_name = get_subdomain_name(form) + + instance = form.save(commit=False) + + # Handle status creation or retrieval + status = get_or_create_status(instance, app_id) + + # Retrieve or create the subdomain + subdomain, created = Subdomain.objects.get_or_create(subdomain=subdomain_name, project=project) + + if app_id: + handle_subdomain_change(instance, subdomain, subdomain_name) + + app_slug = handle_shiny_proxy_case(instance, app_slug, app_id) + + app = get_app(app_slug) + + setup_instance(instance, subdomain, app, project, status) + save_instance_and_related_data(instance, form) + + deploy_resource.delay(instance.serialize()) + + +def get_subdomain_name(form): + subdomain_name = form.cleaned_data.get("subdomain") + if not subdomain_name: + raise ValueError("Subdomain is required") + return subdomain_name + + +def get_or_create_status(instance, app_id): + return instance.app_status if app_id else AppStatus.objects.create() + + +def handle_subdomain_change(instance, subdomain, subdomain_name): + from .tasks import delete_resource + + if instance.subdomain.subdomain != subdomain_name: + # In this special case, we avoid async task. + delete_resource(instance.serialize()) + old_subdomain = instance.subdomain + instance.subdomain = subdomain + instance.save(update_fields=["subdomain"]) + if old_subdomain: + old_subdomain.delete() + + +def handle_shiny_proxy_case(instance, app_slug, app_id): + conditions = {("shinyapp", True): "shinyproxyapp", ("shinyproxyapp", False): "shinyapp"} + + proxy_status = getattr(instance, "proxy", False) + new_slug = conditions.get((app_slug, proxy_status), app_slug) + + return new_slug + + +def get_app(app_slug): + try: + return Apps.objects.get(slug=app_slug) + except Apps.DoesNotExist: + logger.error("App with slug %s not found during instance creation", app_slug) + raise ValueError(f"App with slug {app_slug} not found") + + +def setup_instance(instance, subdomain, app, project, status): + instance.subdomain = subdomain + instance.app = app + instance.chart = instance.app.chart + instance.project = project + instance.owner = project.owner + instance.app_status = status + + +def save_instance_and_related_data(instance, form): + instance.save() + form.save_m2m() + instance.set_k8s_values() + instance.url = get_URI(instance.k8s_values) + instance.save(update_fields=["k8s_values", "url"]) diff --git a/apps/migrations/0001_initial.py b/apps/migrations/0001_initial.py new file mode 100644 index 000000000..ff81eb171 --- /dev/null +++ b/apps/migrations/0001_initial.py @@ -0,0 +1,211 @@ +# Generated by Django 5.0.2 on 2024-05-27 07:28 + +import django.db.models.deletion +import tagulous.models.models +from django.db import migrations, models + + +class Migration(migrations.Migration): + initial = True + + dependencies = [] + + operations = [ + migrations.CreateModel( + name="AppCategories", + fields=[ + ("name", models.CharField(max_length=512)), + ("priority", models.IntegerField(default=100)), + ("slug", models.CharField(default="", max_length=512, primary_key=True, serialize=False)), + ], + options={ + "verbose_name": "App Category", + "verbose_name_plural": "App Categories", + }, + ), + migrations.CreateModel( + name="AppStatus", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("info", models.JSONField(blank=True, null=True)), + ("status", models.CharField(default="Creating", max_length=15)), + ("time", models.DateTimeField(auto_now_add=True)), + ], + options={ + "verbose_name": "App Status", + "verbose_name_plural": "App Statuses", + "get_latest_by": "time", + }, + ), + migrations.CreateModel( + name="BaseAppInstance", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("chart", models.CharField(max_length=512)), + ("created_on", models.DateTimeField(auto_now_add=True)), + ("deleted_on", models.DateTimeField(blank=True, null=True)), + ("info", models.JSONField(blank=True, null=True)), + ("name", models.CharField(default="app_name", max_length=512)), + ("k8s_values", models.JSONField(blank=True, null=True)), + ("url", models.URLField(blank=True, null=True)), + ("updated_on", models.DateTimeField(auto_now=True)), + ], + options={ + "verbose_name": "# BASE APP INSTANCE", + "verbose_name_plural": "# BASE APP INSTANCES", + "permissions": [("can_access_app", "Can access app service")], + }, + ), + migrations.CreateModel( + name="Subdomain", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("created_at", models.DateTimeField(auto_now_add=True)), + ("subdomain", models.CharField(max_length=53, unique=True)), + ], + options={ + "verbose_name": "Subdomain", + "verbose_name_plural": "Subdomains", + }, + ), + migrations.CreateModel( + name="Tagulous_CustomAppInstance_tags", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), + ], + options={ + "ordering": ("name",), + "abstract": False, + }, + bases=(tagulous.models.models.BaseTagModel, models.Model), + ), + migrations.CreateModel( + name="Tagulous_DashInstance_tags", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), + ], + options={ + "ordering": ("name",), + "abstract": False, + }, + bases=(tagulous.models.models.BaseTagModel, models.Model), + ), + migrations.CreateModel( + name="Tagulous_ShinyInstance_tags", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), + ], + options={ + "ordering": ("name",), + "abstract": False, + }, + bases=(tagulous.models.models.BaseTagModel, models.Model), + ), + migrations.CreateModel( + name="Tagulous_SocialMixin_tags", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), + ], + options={ + "ordering": ("name",), + "abstract": False, + }, + bases=(tagulous.models.models.BaseTagModel, models.Model), + ), + migrations.CreateModel( + name="Tagulous_TissuumapsInstance_tags", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), + ], + options={ + "ordering": ("name",), + "abstract": False, + }, + bases=(tagulous.models.models.BaseTagModel, models.Model), + ), + migrations.CreateModel( + name="Apps", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("user_can_create", models.BooleanField(default=True)), + ("user_can_edit", models.BooleanField(default=True)), + ("user_can_delete", models.BooleanField(default=True)), + ("access", models.CharField(blank=True, default="public", max_length=20, null=True)), + ("chart", models.CharField(max_length=512)), + ("chart_archive", models.FileField(blank=True, null=True, upload_to="apps/")), + ("created_on", models.DateTimeField(auto_now_add=True)), + ("description", models.TextField(blank=True, default="", null=True)), + ("logo", models.CharField(blank=True, max_length=512, null=True)), + ("name", models.CharField(max_length=512)), + ("priority", models.IntegerField(default=100)), + ("revision", models.IntegerField(default=1)), + ("settings", models.JSONField(blank=True, null=True)), + ("slug", models.CharField(blank=True, max_length=512, null=True)), + ("table_field", models.JSONField(blank=True, null=True)), + ("updated_on", models.DateTimeField(auto_now=True)), + ( + "category", + models.ForeignKey( + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="apps", + to="apps.appcategories", + ), + ), + ], + options={ + "verbose_name": "App Template", + "verbose_name_plural": "App Templates", + }, + ), + ] diff --git a/apps/migrations/0002_initial.py b/apps/migrations/0002_initial.py new file mode 100644 index 000000000..377cb3dd3 --- /dev/null +++ b/apps/migrations/0002_initial.py @@ -0,0 +1,526 @@ +# Generated by Django 5.0.2 on 2024-05-27 07:28 + +import django.core.validators +import django.db.models.deletion +import tagulous.models.fields +from django.conf import settings +from django.db import migrations, models + + +class Migration(migrations.Migration): + initial = True + + dependencies = [ + ("apps", "0001_initial"), + ("portal", "0001_initial"), + ("projects", "0001_initial"), + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ] + + operations = [ + migrations.AddField( + model_name="apps", + name="projects", + field=models.ManyToManyField(blank=True, to="projects.project"), + ), + migrations.CreateModel( + name="NetpolicyInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ], + options={ + "verbose_name": "Network Policy", + "verbose_name_plural": "Network Policies", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + migrations.CreateModel( + name="VolumeInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "size", + models.IntegerField( + default=1, + help_text="Size in GB", + validators=[ + django.core.validators.MinValueValidator(1), + django.core.validators.MaxValueValidator(100), + ], + ), + ), + ], + options={ + "verbose_name": "Persistent Volume", + "verbose_name_plural": "Persistent Volumes", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + migrations.AddField( + model_name="baseappinstance", + name="app", + field=models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="%(app_label)s_%(class)s_related", + to="apps.apps", + ), + ), + migrations.AddField( + model_name="baseappinstance", + name="app_status", + field=models.OneToOneField( + null=True, on_delete=django.db.models.deletion.RESTRICT, related_name="%(class)s", to="apps.appstatus" + ), + ), + migrations.AddField( + model_name="baseappinstance", + name="flavor", + field=models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.RESTRICT, + related_name="%(class)s", + to="projects.flavor", + ), + ), + migrations.AddField( + model_name="baseappinstance", + name="owner", + field=models.ForeignKey( + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="%(class)s", + to=settings.AUTH_USER_MODEL, + ), + ), + migrations.AddField( + model_name="baseappinstance", + name="project", + field=models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, related_name="%(class)s", to="projects.project" + ), + ), + migrations.AddField( + model_name="subdomain", + name="project", + field=models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project"), + ), + migrations.AddField( + model_name="baseappinstance", + name="subdomain", + field=models.OneToOneField( + on_delete=django.db.models.deletion.CASCADE, related_name="%(class)s", to="apps.subdomain" + ), + ), + migrations.AlterUniqueTogether( + name="tagulous_customappinstance_tags", + unique_together={("slug",)}, + ), + migrations.AlterUniqueTogether( + name="tagulous_dashinstance_tags", + unique_together={("slug",)}, + ), + migrations.AlterUniqueTogether( + name="tagulous_shinyinstance_tags", + unique_together={("slug",)}, + ), + migrations.AlterUniqueTogether( + name="tagulous_socialmixin_tags", + unique_together={("slug",)}, + ), + migrations.AlterUniqueTogether( + name="tagulous_tissuumapsinstance_tags", + unique_together={("slug",)}, + ), + migrations.AlterUniqueTogether( + name="apps", + unique_together={("slug", "revision")}, + ), + migrations.CreateModel( + name="DashInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "logs_enabled", + models.BooleanField( + default=True, help_text="Indicates whether logs are activated and visible to the user" + ), + ), + ("note_on_linkonly_privacy", models.TextField(blank=True, default="", null=True)), + ("source_code_url", models.URLField(blank=True, null=True)), + ("description", models.TextField(blank=True, default="", null=True)), + ( + "access", + models.CharField( + choices=[ + ("project", "Project"), + ("private", "Private"), + ("public", "Public"), + ("link", "Link"), + ], + default="private", + max_length=20, + ), + ), + ("port", models.IntegerField(default=8000)), + ("image", models.CharField(max_length=255)), + ("collections", models.ManyToManyField(blank=True, related_name="%(class)s", to="portal.collection")), + ( + "tags", + tagulous.models.fields.TagField( + _set_tag_meta=True, + blank=True, + help_text="Enter a comma-separated tag string", + to="apps.tagulous_dashinstance_tags", + ), + ), + ], + options={ + "verbose_name": "Dash App Instance", + "verbose_name_plural": "Dash App Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance", models.Model), + ), + migrations.CreateModel( + name="ShinyInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "logs_enabled", + models.BooleanField( + default=True, help_text="Indicates whether logs are activated and visible to the user" + ), + ), + ("note_on_linkonly_privacy", models.TextField(blank=True, default="", null=True)), + ("source_code_url", models.URLField(blank=True, null=True)), + ("description", models.TextField(blank=True, default="", null=True)), + ( + "access", + models.CharField( + choices=[ + ("project", "Project"), + ("private", "Private"), + ("public", "Public"), + ("link", "Link"), + ], + default="private", + max_length=20, + ), + ), + ("port", models.IntegerField(default=3838)), + ("image", models.CharField(max_length=255)), + ("proxy", models.BooleanField(default=True)), + ("container_waittime", models.IntegerField(default=20000)), + ("heartbeat_timeout", models.IntegerField(default=60000)), + ("heartbeat_rate", models.IntegerField(default=10000)), + ("collections", models.ManyToManyField(blank=True, related_name="%(class)s", to="portal.collection")), + ( + "tags", + tagulous.models.fields.TagField( + _set_tag_meta=True, + blank=True, + help_text="Enter a comma-separated tag string", + to="apps.tagulous_shinyinstance_tags", + ), + ), + ], + options={ + "verbose_name": "Shiny App Instance", + "verbose_name_plural": "Shiny App Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance", models.Model), + ), + migrations.CreateModel( + name="TissuumapsInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "logs_enabled", + models.BooleanField( + default=True, help_text="Indicates whether logs are activated and visible to the user" + ), + ), + ("note_on_linkonly_privacy", models.TextField(blank=True, default="", null=True)), + ("source_code_url", models.URLField(blank=True, null=True)), + ("description", models.TextField(blank=True, default="", null=True)), + ( + "access", + models.CharField( + choices=[ + ("project", "Project"), + ("private", "Private"), + ("public", "Public"), + ("link", "Link"), + ], + default="private", + max_length=20, + ), + ), + ("collections", models.ManyToManyField(blank=True, related_name="%(class)s", to="portal.collection")), + ( + "tags", + tagulous.models.fields.TagField( + _set_tag_meta=True, + blank=True, + help_text="Enter a comma-separated tag string", + to="apps.tagulous_tissuumapsinstance_tags", + ), + ), + ( + "volume", + models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="%(class)s", + to="apps.volumeinstance", + ), + ), + ], + options={ + "verbose_name": "TissUUmaps Instance", + "verbose_name_plural": "TissUUmaps Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance", models.Model), + ), + migrations.CreateModel( + name="RStudioInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "access", + models.CharField( + choices=[("project", "Project"), ("private", "Private")], default="private", max_length=20 + ), + ), + ("volume", models.ManyToManyField(blank=True, to="apps.volumeinstance")), + ], + options={ + "verbose_name": "RStudio Instance", + "verbose_name_plural": "RStudio Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + migrations.CreateModel( + name="JupyterInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "access", + models.CharField( + choices=[("project", "Project"), ("private", "Private")], default="private", max_length=20 + ), + ), + ("volume", models.ManyToManyField(blank=True, to="apps.volumeinstance")), + ], + options={ + "verbose_name": "JupyterLab Instance", + "verbose_name_plural": "JupyterLab Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + migrations.CreateModel( + name="FilemanagerInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "access", + models.CharField( + choices=[("project", "Project"), ("private", "Private")], default="project", max_length=20 + ), + ), + ("persistent", models.BooleanField(default=False)), + ("volume", models.ManyToManyField(blank=True, to="apps.volumeinstance")), + ], + options={ + "verbose_name": "Filemanager", + "verbose_name_plural": "Filemanagers", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + migrations.CreateModel( + name="CustomAppInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "logs_enabled", + models.BooleanField( + default=True, help_text="Indicates whether logs are activated and visible to the user" + ), + ), + ("note_on_linkonly_privacy", models.TextField(blank=True, default="", null=True)), + ("source_code_url", models.URLField(blank=True, null=True)), + ("description", models.TextField(blank=True, default="", null=True)), + ( + "access", + models.CharField( + choices=[ + ("project", "Project"), + ("private", "Private"), + ("public", "Public"), + ("link", "Link"), + ], + default="private", + max_length=20, + ), + ), + ("port", models.IntegerField(default=8000)), + ("image", models.CharField(max_length=255)), + ("path", models.CharField(default="/", max_length=255)), + ("user_id", models.IntegerField(default=1000)), + ("collections", models.ManyToManyField(blank=True, related_name="%(class)s", to="portal.collection")), + ( + "tags", + tagulous.models.fields.TagField( + _set_tag_meta=True, + blank=True, + help_text="Enter a comma-separated tag string", + to="apps.tagulous_customappinstance_tags", + ), + ), + ( + "volume", + models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="%(class)s", + to="apps.volumeinstance", + ), + ), + ], + options={ + "verbose_name": "Custom App Instance", + "verbose_name_plural": "Custom App Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance", models.Model), + ), + migrations.CreateModel( + name="VSCodeInstance", + fields=[ + ( + "baseappinstance_ptr", + models.OneToOneField( + auto_created=True, + on_delete=django.db.models.deletion.CASCADE, + parent_link=True, + primary_key=True, + serialize=False, + to="apps.baseappinstance", + ), + ), + ( + "access", + models.CharField( + choices=[("project", "Project"), ("private", "Private")], default="private", max_length=20 + ), + ), + ("volume", models.ManyToManyField(blank=True, to="apps.volumeinstance")), + ], + options={ + "verbose_name": "VS Code Instance", + "verbose_name_plural": "VS Code Instances", + "permissions": [("can_access_app", "Can access app service")], + }, + bases=("apps.baseappinstance",), + ), + ] diff --git a/apps/models.py b/apps/models.py deleted file mode 100644 index bd16ab76e..000000000 --- a/apps/models.py +++ /dev/null @@ -1,223 +0,0 @@ -from datetime import datetime, timedelta - -from django.conf import settings -from django.contrib.auth import get_user_model -from django.db import models -from django.db.models import Q -from django.db.models.signals import post_save -from django.dispatch import receiver -from guardian.shortcuts import assign_perm, remove_perm -from tagulous.models import TagField - - -class AppCategories(models.Model): - name = models.CharField(max_length=512) - priority = models.IntegerField(default=100) - slug = models.CharField(max_length=512, default="", primary_key=True) - - def __str__(self): - return str(self.name) - - -class Apps(models.Model): - user_can_create = models.BooleanField(default=True) - user_can_edit = models.BooleanField(default=True) - user_can_delete = models.BooleanField(default=True) - access = models.CharField(max_length=20, blank=True, null=True, default="public") - category = models.ForeignKey( - "AppCategories", - related_name="apps", - on_delete=models.CASCADE, - null=True, - ) - chart = models.CharField(max_length=512) - chart_archive = models.FileField(upload_to="apps/", null=True, blank=True) - created_on = models.DateTimeField(auto_now_add=True) - description = models.TextField(blank=True, null=True, default="") - logo = models.CharField(max_length=512, null=True, blank=True) - name = models.CharField(max_length=512) - priority = models.IntegerField(default=100) - projects = models.ManyToManyField("projects.Project", blank=True) - revision = models.IntegerField(default=1) - settings = models.JSONField(blank=True, null=True) - slug = models.CharField(max_length=512, blank=True, null=True) - table_field = models.JSONField(blank=True, null=True) - updated_on = models.DateTimeField(auto_now=True) - - class Meta: - unique_together = ( - "slug", - "revision", - ) - - def __str__(self): - return str(self.name) + "({})".format(self.revision) - - -class AppInstanceManager(models.Manager): - def get_app_instances_of_project_filter(self, user, project, include_deleted=False, deleted_time_delta=None): - q = Q() - - if not include_deleted: - if deleted_time_delta is None: - q &= ~Q(state="Deleted") - else: - time_threshold = datetime.now() - timedelta(minutes=deleted_time_delta) - q &= ~Q(state="Deleted") | Q(deleted_on__gte=time_threshold) - - q &= Q(owner=user) | Q( - access__in=["project", "public", "private", "link"] if user.is_superuser else ["project", "public", "link"] - ) - q &= Q(project=project) - - return q - - def get_app_instances_of_project( - self, - user, - project, - filter_func=None, - order_by=None, - limit=None, - override_default_filter=False, - ): - if order_by is None: - order_by = "-created_on" - - if filter_func is None: - return self.filter(self.get_app_instances_of_project_filter(user=user, project=project)).order_by(order_by)[ - :limit - ] - - if override_default_filter: - return self.filter(filter_func).order_by(order_by)[:limit] - - return ( - self.filter(self.get_app_instances_of_project_filter(user=user, project=project)) - .filter(filter_func) - .order_by(order_by)[:limit] - ) - - def get_available_app_dependencies(self, user, project, app_name): - result = self.filter( - ~Q(state="Deleted"), - Q(owner=user) | Q(access__in=["project", "public"]), - project=project, - app__name=app_name, - ) - - if settings.STUDIO_ACCESSMODE == "ReadWriteOnce" and app_name == "Persistent Volume": - for instance in result: - exists = self.filter( - ~Q(state="Deleted"), - project=project, - app_dependencies=instance, - ).exists() - - if exists: - result = result.exclude(id=instance.id) - - return result - - def user_can_create(self, user, project, app_slug): - apps_per_project = {} if project.apps_per_project is None else project.apps_per_project - - limit = apps_per_project[app_slug] if app_slug in apps_per_project else None - - app = Apps.objects.get(slug=app_slug) - - if not app.user_can_create: - return False - - num_of_app_instances = self.filter( - ~Q(state="Deleted"), - app__slug=app_slug, - project=project, - ).count() - - has_perm = user.has_perm("apps.add_appinstance") - - return limit is None or limit > num_of_app_instances or has_perm - - -class AppInstance(models.Model): - objects = AppInstanceManager() - - access = models.CharField(max_length=20, default="private", null=True, blank=True) - app = models.ForeignKey("Apps", on_delete=models.CASCADE, related_name="appinstance") - app_dependencies = models.ManyToManyField("apps.AppInstance", blank=True) - created_on = models.DateTimeField(auto_now_add=True) - deleted_on = models.DateTimeField(null=True, blank=True) - description = models.TextField(blank=True, null=True, default="") - info = models.JSONField(blank=True, null=True) - model_dependencies = models.ManyToManyField("models.Model", blank=True) - name = models.CharField(max_length=512, default="app_name") - owner = models.ForeignKey( - get_user_model(), - on_delete=models.CASCADE, - related_name="app_owner", - null=True, - ) - parameters = models.JSONField(blank=True, null=True) - project = models.ForeignKey( - "projects.Project", - on_delete=models.CASCADE, - related_name="appinstance", - ) - flavor = models.ForeignKey( - "projects.Flavor", - on_delete=models.RESTRICT, - related_name="appinstance", - null=True, - ) - state = models.CharField(max_length=50, null=True, blank=True) - table_field = models.JSONField(blank=True, null=True) - tags = TagField(blank=True) - updated_on = models.DateTimeField(auto_now=True) - note_on_linkonly_privacy = models.TextField(blank=True, null=True, default="") - collections = models.ManyToManyField("collections_module.Collection", blank=True, related_name="app_instances") - source_code_url = models.URLField(blank=True, null=True) - - class Meta: - permissions = [("can_access_app", "Can access app service")] - - def __str__(self): - return str(self.name) + " ({})-{}-{}-{}".format(self.state, self.owner, self.app.name, self.project) - - -@receiver( - post_save, - sender=AppInstance, - dispatch_uid="app_instance_update_permission", -) -def update_permission(sender, instance, created, **kwargs): - owner = instance.owner - - if instance.access == "private": - if created or not owner.has_perm("can_access_app", instance): - assign_perm("can_access_app", owner, instance) - - else: - if owner.has_perm("can_access_app", instance): - remove_perm("can_access_app", owner, instance) - - -class AppStatus(models.Model): - appinstance = models.ForeignKey("AppInstance", on_delete=models.CASCADE, related_name="status") - info = models.JSONField(blank=True, null=True) - status_type = models.CharField(max_length=15, default="app_name") - time = models.DateTimeField(auto_now_add=True) - - class Meta: - get_latest_by = "time" - - def __str__(self): - return str(self.appinstance.name) + "({})".format(self.time) - - -class ResourceData(models.Model): - appinstance = models.ForeignKey("AppInstance", on_delete=models.CASCADE, related_name="resourcedata") - cpu = models.IntegerField() - gpu = models.IntegerField() - mem = models.IntegerField() - time = models.IntegerField() diff --git a/apps/models/__init__.py b/apps/models/__init__.py new file mode 100644 index 000000000..657371d7c --- /dev/null +++ b/apps/models/__init__.py @@ -0,0 +1,4 @@ +# flake8: noqa: F403 + +from .base import * # isort:skip +from .app_types import * # isort:skip diff --git a/apps/models/app_types/__init__.py b/apps/models/app_types/__init__.py new file mode 100644 index 000000000..7fd820431 --- /dev/null +++ b/apps/models/app_types/__init__.py @@ -0,0 +1,10 @@ +from .custom import CustomAppInstance, CustomAppInstanceManager +from .dash import DashInstance, DashInstanceManager +from .filemanager import FilemanagerInstance, FilemanagerInstanceManager +from .jupyter import JupyterInstance, JupyterInstanceManager +from .netpolicy import NetpolicyInstance, NetpolicyInstanceManager +from .rstudio import RStudioInstance, RStudioInstanceManager +from .shiny import ShinyInstance, ShinyInstanceManager +from .tissuumaps import TissuumapsInstance, TissuumapsInstanceManager +from .volume import VolumeInstance, VolumeInstanceManager +from .vscode import VSCodeInstance, VSCodeInstanceManager diff --git a/apps/models/app_types/custom.py b/apps/models/app_types/custom.py new file mode 100644 index 000000000..cc6bb8edb --- /dev/null +++ b/apps/models/app_types/custom.py @@ -0,0 +1,50 @@ +from django.db import models + +from apps.models import ( + AppInstanceManager, + BaseAppInstance, + LogsEnabledMixin, + SocialMixin, +) + + +class CustomAppInstanceManager(AppInstanceManager): + model_type = "customappinstance" + + +class CustomAppInstance(BaseAppInstance, SocialMixin, LogsEnabledMixin): + objects = CustomAppInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ( + "private", + "Private", + ), + ("public", "Public"), + ("link", "Link"), + ) + + volume = models.ForeignKey( + "VolumeInstance", blank=True, null=True, related_name="%(class)s", on_delete=models.CASCADE + ) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + port = models.IntegerField(default=8000) + image = models.CharField(max_length=255) + path = models.CharField(max_length=255, default="/") + user_id = models.IntegerField(default=1000) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + k8s_values["appconfig"] = dict(port=self.port, image=self.image, path=self.path, userid=self.user_id) + volumeK8s_dict = {"volumeK8s": {}} + if self.volume: + volumeK8s_dict["volumeK8s"][self.volume.name] = dict(release=self.volume.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + return k8s_values + + class Meta: + verbose_name = "Custom App Instance" + verbose_name_plural = "Custom App Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/dash.py b/apps/models/app_types/dash.py new file mode 100644 index 000000000..b7e114bc1 --- /dev/null +++ b/apps/models/app_types/dash.py @@ -0,0 +1,40 @@ +from django.db import models + +from apps.models import ( + AppInstanceManager, + BaseAppInstance, + LogsEnabledMixin, + SocialMixin, +) + + +class DashInstanceManager(AppInstanceManager): + model_type = "dashinstance" + + +class DashInstance(BaseAppInstance, SocialMixin, LogsEnabledMixin): + objects = DashInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ( + "private", + "Private", + ), + ("public", "Public"), + ("link", "Link"), + ) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + port = models.IntegerField(default=8000) + image = models.CharField(max_length=255) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + k8s_values["appconfig"] = dict(port=self.port, image=self.image) + return k8s_values + + class Meta: + verbose_name = "Dash App Instance" + verbose_name_plural = "Dash App Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/filemanager.py b/apps/models/app_types/filemanager.py new file mode 100644 index 000000000..2bd1a32a8 --- /dev/null +++ b/apps/models/app_types/filemanager.py @@ -0,0 +1,40 @@ +from django.db import models + +from apps.models import AppInstanceManager, BaseAppInstance + + +class FilemanagerInstanceManager(AppInstanceManager): + model_type = "filemanagerinstance" + + +class FilemanagerInstance(BaseAppInstance): + objects = FilemanagerInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ("private", "Private"), + ) + volume = models.ManyToManyField("VolumeInstance", blank=True) + access = models.CharField(max_length=20, default="project", choices=ACCESS_TYPES) + persistent = models.BooleanField(default=False) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + + # Not the nicest perhaps, but it works since the charts assume that the volumes are on this form + # {apps: + # {volumeK8s: + # {project-vol: + # {release: r1582t9h9 + + volumeK8s_dict = {"volumeK8s": {}} + for object in self.volume.all(): + volumeK8s_dict["volumeK8s"][object.name] = dict(release=object.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + return k8s_values + + class Meta: + verbose_name = "Filemanager" + verbose_name_plural = "Filemanagers" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/jupyter.py b/apps/models/app_types/jupyter.py new file mode 100644 index 000000000..b9b2c55a0 --- /dev/null +++ b/apps/models/app_types/jupyter.py @@ -0,0 +1,41 @@ +from django.db import models + +from apps.models import AppInstanceManager, BaseAppInstance + + +class JupyterInstanceManager(AppInstanceManager): + model_type = "jupyterinstance" + + +class JupyterInstance(BaseAppInstance): + objects = JupyterInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ("private", "Private"), + ) + volume = models.ManyToManyField("VolumeInstance", blank=True) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + # Not the nicest perhaps, but it works since the charts assume that the volumes are on this form + # {apps: + # {volumeK8s: + # {project-vol: + # {release: r1582t9h9 + + volumeK8s_dict = {"volumeK8s": {}} + for object in self.volume.all(): + volumeK8s_dict["volumeK8s"][object.name] = dict(release=object.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + # This is just do fix a legacy. + # TODO: Change the jupyter chart to fetch port from appconfig as other apps + k8s_values["service"]["targetport"] = 8888 + return k8s_values + + class Meta: + verbose_name = "JupyterLab Instance" + verbose_name_plural = "JupyterLab Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/netpolicy.py b/apps/models/app_types/netpolicy.py new file mode 100644 index 000000000..ab158a9c4 --- /dev/null +++ b/apps/models/app_types/netpolicy.py @@ -0,0 +1,36 @@ +from datetime import datetime, timedelta + +from django.db.models import Q + +from apps.models import AppInstanceManager, BaseAppInstance + + +class NetpolicyInstanceManager(AppInstanceManager): + model_type = "netpolicyinstance" + + def get_app_instances_of_project_filter(self, user, project, include_deleted=False, deleted_time_delta=None): + q = Q() + + if not include_deleted: + if deleted_time_delta is None: + q &= ~Q(app_status__status="Deleted") + else: + time_threshold = datetime.now() - timedelta(minutes=deleted_time_delta) + q &= ~Q(app_status__status="Deleted") | Q(deleted_on__gte=time_threshold) + + q &= Q(owner=user) + q &= Q(project=project) + + return q + + +class NetpolicyInstance(BaseAppInstance): + objects = NetpolicyInstanceManager() + + def __str__(self): + return str(self.name) + + class Meta: + verbose_name = "Network Policy" + verbose_name_plural = "Network Policies" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/rstudio.py b/apps/models/app_types/rstudio.py new file mode 100644 index 000000000..f23a84920 --- /dev/null +++ b/apps/models/app_types/rstudio.py @@ -0,0 +1,42 @@ +from django.db import models + +from apps.models import AppInstanceManager, BaseAppInstance + + +class RStudioInstanceManager(AppInstanceManager): + model_type = "rstudioinstance" + + +class RStudioInstance(BaseAppInstance): + objects = RStudioInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ("private", "Private"), + ) + volume = models.ManyToManyField("VolumeInstance", blank=True) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + # Not the nicest perhaps, but it works since the charts assume that the volumes are on this form + # {apps: + # {volumeK8s: + # {project-vol: + # {release: r1582t9h9 + + volumeK8s_dict = {"volumeK8s": {}} + for object in self.volume.all(): + volumeK8s_dict["volumeK8s"][object.name] = dict(release=object.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + + # This is just do fix a legacy. + # TODO: Change the rstdio chart to fetch port from appconfig as other apps + k8s_values["service"]["targetport"] = 8787 + return k8s_values + + class Meta: + verbose_name = "RStudio Instance" + verbose_name_plural = "RStudio Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/shiny.py b/apps/models/app_types/shiny.py new file mode 100644 index 000000000..450c7e942 --- /dev/null +++ b/apps/models/app_types/shiny.py @@ -0,0 +1,50 @@ +from django.db import models + +from apps.models import ( + AppInstanceManager, + BaseAppInstance, + LogsEnabledMixin, + SocialMixin, +) + + +class ShinyInstanceManager(AppInstanceManager): + model_type = "shinyinstance" + + +class ShinyInstance(BaseAppInstance, SocialMixin, LogsEnabledMixin): + objects = ShinyInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ( + "private", + "Private", + ), + ("public", "Public"), + ("link", "Link"), + ) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + port = models.IntegerField(default=3838) + image = models.CharField(max_length=255) + proxy = models.BooleanField(default=True) + container_waittime = models.IntegerField(default=20000) + heartbeat_timeout = models.IntegerField(default=60000) + heartbeat_rate = models.IntegerField(default=10000) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + k8s_values["appconfig"] = dict( + port=self.port, + image=self.image, + proxyheartbeatrate=self.heartbeat_rate, + proxyheartbeattimeout=self.heartbeat_timeout, + proxycontainerwaittime=self.container_waittime, + ) + return k8s_values + + class Meta: + verbose_name = "Shiny App Instance" + verbose_name_plural = "Shiny App Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/tissuumaps.py b/apps/models/app_types/tissuumaps.py new file mode 100644 index 000000000..0554c6562 --- /dev/null +++ b/apps/models/app_types/tissuumaps.py @@ -0,0 +1,50 @@ +from django.db import models + +from apps.models import ( + AppInstanceManager, + BaseAppInstance, + LogsEnabledMixin, + SocialMixin, +) + + +class TissuumapsInstanceManager(AppInstanceManager): + model_type = "shinyproxyinstance" + + +class TissuumapsInstance(BaseAppInstance, SocialMixin, LogsEnabledMixin): + objects = TissuumapsInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ( + "private", + "Private", + ), + ("public", "Public"), + ("link", "Link"), + ) + volume = models.ForeignKey( + "VolumeInstance", blank=True, null=True, related_name="%(class)s", on_delete=models.CASCADE + ) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + # Not the nicest perhaps, but it works since the charts assume that the volumes are on this form + # {apps: + # {volumeK8s: + # {project-vol: + # {release: r1582t9h9 + + volumeK8s_dict = {"volumeK8s": {}} + if self.volume: + volumeK8s_dict["volumeK8s"][self.volume.name] = dict(release=self.volume.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + return k8s_values + + class Meta: + verbose_name = "TissUUmaps Instance" + verbose_name_plural = "TissUUmaps Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/volume.py b/apps/models/app_types/volume.py new file mode 100644 index 000000000..9d7dcea08 --- /dev/null +++ b/apps/models/app_types/volume.py @@ -0,0 +1,31 @@ +from datetime import datetime, timedelta + +from django.core.validators import MaxValueValidator, MinValueValidator +from django.db import models +from django.db.models import Q + +from apps.models import AppInstanceManager, BaseAppInstance + + +class VolumeInstanceManager(AppInstanceManager): + model_type = "volumeinstance" + + +class VolumeInstance(BaseAppInstance): + objects = VolumeInstanceManager() + size = models.IntegerField( + default=1, help_text="Size in GB", validators=[MinValueValidator(1), MaxValueValidator(100)] + ) + + def __str__(self): + return str(self.name) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + k8s_values["volume"] = dict(size=f"{str(self.size)}Gi") + return k8s_values + + class Meta: + verbose_name = "Persistent Volume" + verbose_name_plural = "Persistent Volumes" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/app_types/vscode.py b/apps/models/app_types/vscode.py new file mode 100644 index 000000000..714fe76da --- /dev/null +++ b/apps/models/app_types/vscode.py @@ -0,0 +1,38 @@ +from django.db import models + +from apps.models import AppInstanceManager, BaseAppInstance + + +class VSCodeInstanceManager(AppInstanceManager): + model_type = "vscodeinstance" + + +class VSCodeInstance(BaseAppInstance): + objects = VSCodeInstanceManager() + ACCESS_TYPES = ( + ("project", "Project"), + ("private", "Private"), + ) + volume = models.ManyToManyField("VolumeInstance", blank=True) + access = models.CharField(max_length=20, default="private", choices=ACCESS_TYPES) + + def get_k8s_values(self): + k8s_values = super().get_k8s_values() + + k8s_values["permission"] = str(self.access) + # Not the nicest perhaps, but it works since the charts assume that the volumes are on this form + # {apps: + # {volumeK8s: + # {project-vol: + # {release: r1582t9h9 + + volumeK8s_dict = {"volumeK8s": {}} + for object in self.volume.all(): + volumeK8s_dict["volumeK8s"][object.name] = dict(release=object.subdomain.subdomain) + k8s_values["apps"] = volumeK8s_dict + return k8s_values + + class Meta: + verbose_name = "VS Code Instance" + verbose_name_plural = "VS Code Instances" + permissions = [("can_access_app", "Can access app service")] diff --git a/apps/models/base/__init__.py b/apps/models/base/__init__.py new file mode 100644 index 000000000..4d8e20460 --- /dev/null +++ b/apps/models/base/__init__.py @@ -0,0 +1,7 @@ +from .app_categories import AppCategories +from .app_status import AppStatus +from .app_template import Apps +from .base import AppInstanceManager, BaseAppInstance +from .logs_enabled_mixin import LogsEnabledMixin +from .social_mixin import SocialMixin +from .subdomain import Subdomain diff --git a/apps/models/base/app_categories.py b/apps/models/base/app_categories.py new file mode 100644 index 000000000..25920863e --- /dev/null +++ b/apps/models/base/app_categories.py @@ -0,0 +1,14 @@ +from django.db import models + + +class AppCategories(models.Model): + name = models.CharField(max_length=512) + priority = models.IntegerField(default=100) + slug = models.CharField(max_length=512, default="", primary_key=True) + + def __str__(self): + return str(self.name) + + class Meta: + verbose_name = "App Category" + verbose_name_plural = "App Categories" diff --git a/apps/models/base/app_status.py b/apps/models/base/app_status.py new file mode 100644 index 000000000..13356bc2d --- /dev/null +++ b/apps/models/base/app_status.py @@ -0,0 +1,15 @@ +from django.db import models + + +class AppStatus(models.Model): + info = models.JSONField(blank=True, null=True) + status = models.CharField(max_length=15, default="Creating") + time = models.DateTimeField(auto_now_add=True) + + class Meta: + get_latest_by = "time" + verbose_name = "App Status" + verbose_name_plural = "App Statuses" + + def __str__(self): + return f"{str(self.status)} ({str(self.time)})" diff --git a/apps/models/base/app_template.py b/apps/models/base/app_template.py new file mode 100644 index 000000000..fcde4ab87 --- /dev/null +++ b/apps/models/base/app_template.py @@ -0,0 +1,43 @@ +from django.db import models + + +class Apps(models.Model): + """Essentially app template""" + + user_can_create = models.BooleanField(default=True) + user_can_edit = models.BooleanField(default=True) + user_can_delete = models.BooleanField(default=True) + access = models.CharField(max_length=20, blank=True, null=True, default="public") + category = models.ForeignKey( + "AppCategories", + related_name="apps", + on_delete=models.CASCADE, + null=True, + ) + chart = models.CharField(max_length=512) + chart_archive = models.FileField(upload_to="apps/", null=True, blank=True) + created_on = models.DateTimeField(auto_now_add=True) + description = models.TextField(blank=True, null=True, default="") + logo = models.CharField(max_length=512, null=True, blank=True) + name = models.CharField(max_length=512) + priority = models.IntegerField(default=100) + projects = models.ManyToManyField("projects.Project", blank=True) + revision = models.IntegerField(default=1) + settings = models.JSONField(blank=True, null=True) + slug = models.CharField(max_length=512, blank=True, null=True) + table_field = models.JSONField(blank=True, null=True) + updated_on = models.DateTimeField(auto_now=True) + + class Meta: + unique_together = ( + "slug", + "revision", + ) + verbose_name = "App Template" + verbose_name_plural = "App Templates" + + def __str__(self): + return str(self.name) + "({})".format(self.revision) + + def to_dict(self): + pass diff --git a/apps/models/base/base.py b/apps/models/base/base.py new file mode 100644 index 000000000..382f66726 --- /dev/null +++ b/apps/models/base/base.py @@ -0,0 +1,161 @@ +import json +from datetime import datetime, timedelta + +from django.conf import settings +from django.contrib.auth import get_user_model +from django.core import serializers +from django.db import models +from django.db.models import Q + +from apps.models.base.app_status import AppStatus +from apps.models.base.app_template import Apps +from apps.models.base.subdomain import Subdomain +from projects.models import Flavor, Project + + +class AppInstanceManager(models.Manager): + model_type = "appinstance" + + def get_app_instances_of_project_filter(self, user, project, include_deleted=False, deleted_time_delta=None): + q = Q() + + if not include_deleted: + if deleted_time_delta is None: + q &= ~Q(app_status__status="Deleted") + else: + time_threshold = datetime.now() - timedelta(minutes=deleted_time_delta) + q &= ~Q(app_status__status="Deleted") | Q(deleted_on__gte=time_threshold) + + if hasattr(self.model, "access"): + q &= Q(owner=user) | Q( + access__in=["project", "public", "private", "link"] + if user.is_superuser + else ["project", "public", "link"] + ) + else: + q &= Q(owner=user) + + q &= Q(project=project) + + return q + + def get_app_instances_of_project( + self, + user, + project, + filter_func=None, + order_by=None, + limit=None, + override_default_filter=False, + ): + if order_by is None: + order_by = "-created_on" + + if filter_func is None: + return self.filter(self.get_app_instances_of_project_filter(user=user, project=project)).order_by(order_by)[ + :limit + ] + + if override_default_filter: + return self.filter(filter_func).order_by(order_by)[:limit] + + return ( + self.filter(self.get_app_instances_of_project_filter(user=user, project=project)) + .filter(filter_func) + .order_by(order_by)[:limit] + ) + + def user_can_create(self, user, project, app_slug): + apps_per_project = {} if project.apps_per_project is None else project.apps_per_project + + limit = apps_per_project[app_slug] if app_slug in apps_per_project else None + app = Apps.objects.get(slug=app_slug) + + if not app.user_can_create: + return False + + num_of_app_instances = self.filter( + ~Q(app_status__status="Deleted"), + app__slug=app_slug, + project=project, + ).count() + + has_perm = user.has_perm(f"apps.add_{self.model_type}") + return limit is None or limit > num_of_app_instances or has_perm + + def user_can_edit(self, user, project, app_slug): + app = Apps.objects.get(slug=app_slug) + return app.user_can_edit or user.has_perm(f"apps.change_{self.model_type}") + + +class BaseAppInstance(models.Model): + objects = AppInstanceManager() + + app = models.ForeignKey(Apps, on_delete=models.CASCADE, related_name="%(app_label)s_%(class)s_related") + chart = models.CharField( + max_length=512, + ) + created_on = models.DateTimeField(auto_now_add=True) + deleted_on = models.DateTimeField(null=True, blank=True) + info = models.JSONField(blank=True, null=True) + # model_dependencies = models.ManyToManyField("models.Model", blank=True) + name = models.CharField(max_length=512, default="app_name") + owner = models.ForeignKey( + get_user_model(), + on_delete=models.CASCADE, + related_name="%(class)s", + null=True, + ) + k8s_values = models.JSONField(blank=True, null=True) + project = models.ForeignKey( + Project, + on_delete=models.CASCADE, + related_name="%(class)s", + ) + flavor = models.ForeignKey(Flavor, on_delete=models.RESTRICT, related_name="%(class)s", null=True, blank=True) + subdomain = models.OneToOneField( + Subdomain, + on_delete=models.CASCADE, + related_name="%(class)s", + ) + app_status = models.OneToOneField(AppStatus, on_delete=models.RESTRICT, related_name="%(class)s", null=True) + + url = models.URLField(blank=True, null=True) + updated_on = models.DateTimeField(auto_now=True) + + class Meta: + permissions = [("can_access_app", "Can access app service")] + verbose_name = "# BASE APP INSTANCE" + verbose_name_plural = "# BASE APP INSTANCES" + + def __str__(self): + return f"{self.name}-{self.owner}-{self.app.name}-{self.project}" + + def get_k8s_values(self): + k8s_values = dict( + name=self.name, + appname=self.subdomain.subdomain, + project=dict(name=self.project.name, slug=self.project.slug), + service=dict( + name=self.subdomain.subdomain + "-" + self.app.slug, + ), + **self.subdomain.to_dict(), + **self.flavor.to_dict() if self.flavor else {}, + storageClass=settings.STORAGECLASS, + namespace=settings.NAMESPACE, + release=self.subdomain.subdomain, # This is legacy and should be changed + ) + + # Add global values + k8s_values["global"] = dict( + domain=settings.DOMAIN, + auth_domain=settings.AUTH_DOMAIN, + protocol=settings.AUTH_PROTOCOL, + ) + return k8s_values + + def set_k8s_values(self): + self.k8s_values = self.get_k8s_values() + + def serialize(self): + return json.loads(serializers.serialize("json", [self]))[0] diff --git a/apps/models/base/logs_enabled_mixin.py b/apps/models/base/logs_enabled_mixin.py new file mode 100644 index 000000000..f4d1d9791 --- /dev/null +++ b/apps/models/base/logs_enabled_mixin.py @@ -0,0 +1,10 @@ +from django.db import models + + +class LogsEnabledMixin(models.Model): + logs_enabled = models.BooleanField( + default=True, help_text="Indicates whether logs are activated and visible to the user" + ) + + class Meta: + abstract = True diff --git a/apps/models/base/social_mixin.py b/apps/models/base/social_mixin.py new file mode 100644 index 000000000..034b811e1 --- /dev/null +++ b/apps/models/base/social_mixin.py @@ -0,0 +1,13 @@ +from django.db import models +from tagulous.models import TagField + + +class SocialMixin(models.Model): + tags = TagField(blank=True) + note_on_linkonly_privacy = models.TextField(blank=True, null=True, default="") + collections = models.ManyToManyField("portal.Collection", blank=True, related_name="%(class)s") + source_code_url = models.URLField(blank=True, null=True) + description = models.TextField(blank=True, null=True, default="") + + class Meta: + abstract = True diff --git a/apps/models/base/subdomain.py b/apps/models/base/subdomain.py new file mode 100644 index 000000000..f5c5e6168 --- /dev/null +++ b/apps/models/base/subdomain.py @@ -0,0 +1,20 @@ +from django.conf import settings +from django.db import models + + +class Subdomain(models.Model): + created_at = models.DateTimeField(auto_now_add=True) + subdomain = models.CharField(max_length=53, unique=True) + project = models.ForeignKey(settings.PROJECTS_MODEL, on_delete=models.CASCADE, null=True) + + def __str__(self): + return str(self.subdomain) + " ({})".format(self.project.name) + + def to_dict(self): + return { + "subdomain": self.subdomain, + } + + class Meta: + verbose_name = "Subdomain" + verbose_name_plural = "Subdomains" diff --git a/apps/serialize.py b/apps/serialize.py deleted file mode 100644 index 37f07e25a..000000000 --- a/apps/serialize.py +++ /dev/null @@ -1,369 +0,0 @@ -import json - -import flatten_json -from django.conf import settings -from django.contrib.auth import get_user_model -from django.db.models import Q -from django.template import engines -from django.utils.text import slugify -from rest_framework.authtoken.models import Token - -from models.models import Model -from projects.models import S3, Environment, Flavor -from studio.utils import get_logger - -from .models import AppInstance, Apps - -logger = get_logger(__name__) - -User = get_user_model() - -KEYWORDS = [ - "appobj", - "model", - "flavor", - "S3", - "environment", - "volumes", - "apps", - "logs", - "permissions", - "default_values", - "export-cli", - "csrfmiddlewaretoken", - "env_variables", - "publishable", -] - - -def serialize_model(form_selection): - logger.info("SERIALIZING MODEL") - model_json = dict() - obj = [] - if "model" in form_selection: - model_id = form_selection.get("model", None) - if isinstance(model_id, str): - model_id = int(model_id) - obj = Model.objects.filter(pk=model_id) - logger.info("Fetching selected model:") - - object_type = obj[0].object_type.all() - if len(object_type) == 1: - logger.info("OK") - else: - logger.info("Currently only supports one object type per model. Will assume first in list.") - model_json = { - "model": { - "name": obj[0].name, - "version": obj[0].version, - "release_type": obj[0].release_type, - "description": obj[0].description, - "url": "http://{}".format(obj[0].s3.host), - "service": obj[0].s3.app.parameters["service"]["name"], - "port": obj[0].s3.app.parameters["service"]["port"], - "targetport": obj[0].s3.app.parameters["service"]["targetport"], - "access_key": obj[0].s3.access_key, - "secret_key": obj[0].s3.secret_key, - "bucket": obj[0].bucket, - "obj": obj[0].uid, - "path": obj[0].path, - "type": object_type[0].slug, - } - } - - return model_json, obj - - -def serialize_S3(form_selection, project): - logger.info("SERIALIZING S3") - s3_json = dict() - if "S3" in form_selection: - s3_id = form_selection.get("S3", None) - try: - obj = S3.objects.filter(pk=s3_id) - except: # noqa E722 TODO: Add exception - obj = S3.objects.filter(name=s3_id, project=project) - s3_json = { - "s3": { - "pk": obj[0].pk, - "name": obj[0].name, - "host": obj[0].host, - "service": obj[0].app.parameters["service"]["name"], - "port": obj[0].app.parameters["service"]["port"], - "targetport": obj[0].app.parameters["service"]["targetport"], - "access_key": obj[0].access_key, - "secret_key": obj[0].secret_key, - "region": obj[0].region, - } - } - return s3_json - - -def serialize_flavor(form_selection, project): - logger.info("SERIALIZING FLAVOR") - flavor_json = dict() - if "flavor" in form_selection: - flavor_id = form_selection.get("flavor", None) - flavor = Flavor.objects.get(pk=flavor_id, project=project) - flavor_json["flavor"] = { - "requests": { - "cpu": flavor.cpu_req, - "memory": flavor.mem_req, - "ephemeral-storage": flavor.ephmem_req, - }, - "limits": { - "cpu": flavor.cpu_lim, - "memory": flavor.mem_lim, - "ephemeral-storage": flavor.ephmem_lim, - }, - } - if flavor.gpu_req and int(flavor.gpu_req) > 0: - flavor_json["flavor"]["requests"]["nvidia.com/gpu"] = (flavor.gpu_req,) - flavor_json["flavor"]["limits"]["nvidia.com/gpu"] = flavor.gpu_lim - - return flavor_json - - -def serialize_environment(form_selection, project): - logger.info("SERIALIZING ENVIRONMENT") - environment_json = dict() - if "environment" in form_selection: - environment_id = form_selection.get("environment", None) - try: - environment = Environment.objects.get(pk=environment_id) - except: # noqa E722 TODO: Add exception - environment = Environment.objects.get(name=environment_id, project=project) - environment_json["environment"] = { - "pk": environment.pk, - "repository": environment.repository, - "image": environment.image, - "registry": False, - } - if environment.registry: - environment_json["environment"]["registry"] = environment.registry.parameters - environment_json["environment"]["registry"]["enabled"] = True - else: - environment_json["environment"]["registry"] = {"enabled": False} - return environment_json - - -def serialize_apps(form_selection, project): - logger.info("SERIALIZING DEPENDENT APPS") - parameters = dict() - parameters["apps"] = dict() - app_deps = [] - for key in form_selection.keys(): - if "app:" in key and key[0:4] == "app:": - app_name = key[4:] - try: - app = Apps.objects.filter(name=app_name).order_by("-revision").first() - if not app: - app = Apps.objects.filter(slug=app_name).order_by("-revision").first() - except Exception: - logger.error("Failed to fetch app: %s", app_name) - raise - if not app: - logger.info("App not found: %s", app_name) - - parameters["apps"][app.slug] = dict() - logger.info("app: %s id: %s", app_name, str(form_selection[key])) - try: - objs = AppInstance.objects.filter(pk__in=form_selection.getlist(key)) - except: # noqa E722 TODO: Add exception - objs = AppInstance.objects.filter(name__in=form_selection[key], project=project) - - for obj in objs: - app_deps.append(obj) - parameters["apps"][app.slug][slugify(obj.name)] = obj.parameters - - return parameters, app_deps - - -def serialize_primitives(form_selection): - """ - This function serializes all values that are not part of the KEYWORDS list at the top. - """ - logger.info("SERIALIZING PRIMITIVES") - parameters = dict() - # TODO: minor-refactor: do for loop over form_selection without new variable - keys = form_selection.keys() - for key in keys: - if key not in KEYWORDS and "app:" not in key: - parameters[key] = form_selection[key].replace("\r\n", "\n") - - # Turn string booleans to python booleans - if parameters[key] == "False": - parameters[key] = False - elif parameters[key] == "True": - parameters[key] = True - - # Slugify app_name so that users can use special chars in their app names. - elif key == "app_name": - parameters[key] = slugify(parameters[key]) - logger.info(parameters) - - return flatten_json.unflatten(parameters, ".") - - -def serialize_permissions(form_selection): - """ - Serialize permissions from form selection into a dictionary (and later into a JSON object) - - To achieve this we first create a dictionary with all permissions set to False. - Then we set the permission that was selected (or typed in admin panel) to True. - - :param form_selection: form selection from request.POST - :return: dictionary of permissions - """ - logger.info("SERIALIZING PERMISSIONS") - parameters = dict() - parameters["permissions"] = { - "public": False, - "project": False, - "private": False, - "link": False, - } - - permission = form_selection.get("permission", None) - parameters["permissions"][permission] = True - logger.info(parameters) - return parameters - - -def serialize_appobjs(form_selection): - logger.info("SERIALIZING APPOBJS") - parameters = dict() - appobjs = [] - if "appobj" in form_selection: - appobjs = form_selection.getlist("appobj") - parameters["appobj"] = dict() - for obj in appobjs: - app = Apps.objects.get(pk=obj) - parameters["appobj"][app.slug] = True - logger.info(parameters) - return parameters - - -def serialize_default_values(aset): - parameters = [] - if "default_values" in aset: - parameters = dict() - logger.info(aset["default_values"]) - parameters["default_values"] = aset["default_values"] - for key in parameters["default_values"].keys(): - if parameters["default_values"][key] == "False": - parameters["default_values"][key] = False - elif parameters["default_values"][key] == "True": - parameters["default_values"][key] = True - - return parameters - - -def serialize_project(project): - parameters = dict() - if project.mlflow: - parameters["mlflow"] = { - "url": project.mlflow.mlflow_url, - "host": project.mlflow.host, - "service": project.mlflow.app.parameters["service"]["name"], - "port": project.mlflow.app.parameters["service"]["port"], - "targetport": project.mlflow.app.parameters["service"]["targetport"], - "s3url": "https://" + project.mlflow.s3.host, - "s3service": project.mlflow.s3.app.parameters["service"]["name"], - "s3port": project.mlflow.s3.app.parameters["service"]["port"], - "s3targetport": project.mlflow.s3.app.parameters["service"]["targetport"], - "access_key": project.mlflow.s3.access_key, - "secret_key": project.mlflow.s3.secret_key, - "region": project.mlflow.s3.region, - "username": project.mlflow.basic_auth.username, - "password": project.mlflow.basic_auth.password, - } - return parameters - - -def serialize_cli(username, project, aset): - user = User.objects.get(username=username) - token, created = Token.objects.get_or_create(user=user) - parameters = dict() - if "export-cli" in aset and aset["export-cli"] == "True": - parameters["cli_setup"] = { - "url": settings.STUDIO_URL, - "project": project.name, - "user": username, - "token": token.key, - "secure": "False", - } - return parameters - - -def serialize_env_variables(username, project, aset): - logger.info("SERIALIZING ENV VARIABLES") - parameters = dict() - parameters["app_env"] = dict() - logger.info("fetching apps") - try: - apps = AppInstance.objects.filter( - ~Q(state="Deleted"), - Q(owner__username=username) | Q(access__in=["project", "public"]), - project=project, - ) - except Exception as err: - logger.error(err, exc_info=True) - logger.info("Creating template engine") - django_engine = engines["django"] - # TODO: refactor potential bug. If there is an exception thrown by query statement above, then `apps` is not defined - logger.info(apps) - for app in apps: - params = app.parameters - appsettings = app.app.settings - if "env_variables" in appsettings: - tmp = json.dumps(appsettings["env_variables"]) - env_vars = json.loads(django_engine.from_string(tmp).render(params)) - for key in env_vars.keys(): - parameters["app_env"][slugify(key)] = env_vars[key] - logger.info(parameters) - - return parameters - - -def serialize_app(form_selection, project, aset, username): - logger.info("SERIALIZING APP") - parameters = dict() - - model_params, model_deps = serialize_model(form_selection) - parameters.update(model_params) - - app_params, app_deps = serialize_apps(form_selection, project) - parameters.update(app_params) - - prim_params = serialize_primitives(form_selection) - parameters.update(prim_params) - - flavor_params = serialize_flavor(form_selection, project) - parameters.update(flavor_params) - - environment_params = serialize_environment(form_selection, project) - parameters.update(environment_params) - - s3params = serialize_S3(form_selection, project) - parameters.update(s3params) - - permission_params = serialize_permissions(form_selection) - parameters.update(permission_params) - - appobj_params = serialize_appobjs(form_selection) - parameters.update(appobj_params) - - default_values = serialize_default_values(aset) - parameters.update(default_values) - - project_values = serialize_project(project) - parameters.update(project_values) - - cli_values = serialize_cli(username, project, aset) - parameters.update(cli_values) - - env_variables = serialize_env_variables(username, project, aset) - parameters.update(env_variables) - - return parameters, app_deps, model_deps diff --git a/apps/setup.py b/apps/setup.py deleted file mode 100644 index da5fdc98d..000000000 --- a/apps/setup.py +++ /dev/null @@ -1,35 +0,0 @@ -from setuptools import setup - -setup( - name="studio-apps", - version="0.0.1", - description="""Django app for handling portal in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["apps"], - package_dir={"apps": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.1", - "requests==2.31.0", - "django-guardian==2.4.0", - "celery==5.2.7", - "Pillow==9.4.0", - "django-tagulous==1.3.3", - "minio==7.0.2", - "s3fs==2022.1.0", - "flatten-json==0.1.13", - "PyYAML==6.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/apps/signals.py b/apps/signals.py new file mode 100644 index 000000000..5c79d226f --- /dev/null +++ b/apps/signals.py @@ -0,0 +1,71 @@ +from django.db.models.signals import post_save, pre_delete +from django.dispatch import receiver +from guardian.shortcuts import assign_perm, remove_perm + +from apps.app_registry import APP_REGISTRY +from apps.models import BaseAppInstance +from studio.utils import get_logger + +from .tasks import helm_delete + +logger = get_logger(__name__) + + +UID = "app_instance_update_permission" + + +@receiver(pre_delete, sender=BaseAppInstance) +def pre_delete_helm_uninstall(sender, instance, **kwargs): + """ + If object is deleted from the database, then we run helm uninstall + """ + logger.info("PRE DELETING RESOURCES") + + values = instance.k8s_values + if values: + helm_delete.delay(values["subdomain"], values["namespace"]) + else: + logger.error(f"Could not find helm release for {instance}") + + +def update_permission(sender, instance, created, **kwargs): + owner = instance.owner + + access = getattr(instance, "access", None) + + if access is None: + logger.error(f"Access not found in {instance}") + return + + if access == "private": + logger.info(f"Assigning permission to {owner} for {instance}") + if created or not owner.has_perm("can_access_app", instance): + assign_perm("can_access_app", owner, instance) + + else: + if owner.has_perm("can_access_app", instance): + logger.info(f"Removing permission from {owner} for {instance}") + remove_perm("can_access_app", owner, instance) + + +for model in APP_REGISTRY.iter_orm_models(): + receiver(post_save, sender=model, dispatch_uid=UID)(update_permission) + + """ + What is going on here? + Well, after a model is saved, we want to update the permission of the owner of the model. + This signal is triggered after a model is saved, and we update the permission of the owner of the model. + But, since we have many types of models, we must add a reciever for all types of models. + We can do this by iterating over the values of SLUG_MODEL_FORM_MAP, + which is a dictionary that maps the slug of the model to the model itself. + + Equivalent to doing this: + + @receiver(post_save, sender=JupyterInstance, dispatch_uid=UID) + @receiver(post_save, sender=DashInstance, dispatch_uid=UID) + ... + @receiver(post_save, sender=LastModelInstance, dispatch_uid=UID) + def update_perrmission(sender, instance, created, **kwargs): + pass + + """ diff --git a/apps/tasks.py b/apps/tasks.py index b40ba46c3..a9b4fa5c6 100644 --- a/apps/tasks.py +++ b/apps/tasks.py @@ -1,532 +1,162 @@ -import json import subprocess -import time +import uuid +from datetime import datetime -import requests +import yaml from celery import shared_task -from celery.signals import worker_ready from django.apps import apps -from django.conf import settings -from django.core.exceptions import EmptyResultSet +from django.core.exceptions import ObjectDoesNotExist from django.db import transaction -from django.db.models import Q from django.utils import timezone -from models.models import Model, ObjectType -from projects.models import S3, BasicAuth, Environment, MLFlow +from apps.app_registry import APP_REGISTRY from studio.celery import app from studio.utils import get_logger -from . import controller -from .models import AppInstance, Apps, AppStatus, ResourceData +from .models import FilemanagerInstance logger = get_logger(__name__) -K8S_STATUS_MAP = { - "CrashLoopBackOff": "Error", - "Completed": "Retrying...", - "ContainerCreating": "Created", - "PodInitializing": "Pending", -} - -ReleaseName = apps.get_model(app_label=settings.RELEASENAME_MODEL) - - -def get_URI(parameters): - URI = "https://" + parameters["release"] + "." + parameters["global"]["domain"] - - URI = URI.strip("/") - return URI - - -def process_helm_result(results): - stdout = results.stdout.decode("utf-8") - stderr = results.stderr.decode("utf-8") - return stdout, stderr - - -def post_create_hooks(instance): - logger.info("TASK - POST CREATE HOOK...") - # hard coded hooks for now, we can make this dynamic - # and loaded from the app specs - - if instance.app.slug == "minio-admin": - # Create project S3 object - # TODO: If the instance is being updated, - # update the existing S3 object. - access_key = instance.parameters["credentials"]["access_key"] - secret_key = instance.parameters["credentials"]["secret_key"] - - # OBS!! TEMP WORKAROUND to be able to connect to minio - minio_svc = "{}-minio".format(instance.parameters["release"]) - cmd = f"kubectl -n {settings.NAMESPACE} get svc {minio_svc} -o jsonpath='{{.spec.clusterIP}}'" - minio_host_url = "" - try: - result = subprocess.run( - cmd, - shell=True, - capture_output=True, - timeout=settings.KUBE_API_REQUEST_TIMEOUT, - ) - minio_host_url = result.stdout.decode("utf-8") - minio_host_url += ":9000" - except subprocess.CalledProcessError: - logger.error("Oops, something went wrong running the command: %s", cmd) - - try: - s3obj = instance.s3obj - s3obj.access_key = access_key - s3obj.secret_key = secret_key - s3obj.host = minio_host_url - except: # noqa E722 TODO: Add exception - s3obj = S3( - name=instance.name, - project=instance.project, - host=minio_host_url, - access_key=access_key, - secret_key=secret_key, - app=instance, - owner=instance.owner, - ) - s3obj.save() - - if instance.app.slug == "environment": - params = instance.parameters - image = params["container"]["name"] - # We can assume one registry here - for reg_key in params["apps"]["docker_registry"].keys(): - reg_release = params["apps"]["docker_registry"][reg_key]["release"] - reg_domain = params["apps"]["docker_registry"][reg_key]["global"]["domain"] - repository = reg_release + "." + reg_domain - registry = AppInstance.objects.get(parameters__contains={"release": reg_release}) - - target_environment = Environment.objects.get(pk=params["environment"]["pk"]) - target_app = target_environment.app - - env_obj = Environment( - name=instance.name, - project=instance.project, - repository=repository, - image=image, - registry=registry, - app=target_app, - appenv=instance, - ) - env_obj.save() - - if instance.app.slug == "mlflow": - params = instance.parameters - - # OBS!! TEMP WORKAROUND to be able to connect to mlflow (internal dns - # between docker and k8s does not work currently) - # Sure one could use FQDN but lets avoid going via the internet - mlflow_svc = instance.parameters["service"]["name"] - cmd = f"kubectl -n {settings.NAMESPACE} get svc {mlflow_svc} -o jsonpath='{{.spec.clusterIP}}'" - mlflow_host_ip = "" - try: - result = subprocess.run( - cmd, - shell=True, - capture_output=True, - timeout=settings.KUBE_API_REQUEST_TIMEOUT, - ) - mlflow_host_ip = result.stdout.decode("utf-8") - mlflow_host_ip += ":{}".format(instance.parameters["service"]["port"]) - except subprocess.CalledProcessError: - logger.error("Oops, something went wrong running the command: %s", cmd) - - s3 = S3.objects.get(pk=instance.parameters["s3"]["pk"]) - basic_auth = BasicAuth( - owner=instance.owner, - name=instance.name, - project=instance.project, - username=instance.parameters["credentials"]["username"], - password=instance.parameters["credentials"]["password"], - ) - basic_auth.save() - obj = MLFlow( - name=instance.name, - project=instance.project, - mlflow_url="https://{}.{}".format( - instance.parameters["release"], - instance.parameters["global"]["domain"], - ), - s3=s3, - host=mlflow_host_ip, - basic_auth=basic_auth, - app=instance, - owner=instance.owner, - ) - obj.save() - elif instance.app.slug in ("volumeK8s", "netpolicy"): - # Handle volumeK8s and netpolicy creation/recreation - instance.state = "Created" - instance.deleted_on = None - status = AppStatus(appinstance=instance) - status.status_type = "Created" - instance.save() - status.save() - - -def release_name(instance): - # Free up release name (if reserved) - rel_names = instance.releasename_set.all() - - for rel_name in rel_names: - rel_name.status = "active" - rel_name.app = None - rel_name.save() - - -def post_delete_hooks(appinstance): - logger.info("TASK - POST DELETE HOOK...") - release_name(appinstance) - project = appinstance.project - if project.s3storage and project.s3storage.app == appinstance: - project.s3storage.delete() - elif project.mlflow and project.mlflow.app == appinstance: - project.mlflow.delete() - elif appinstance.app.slug in ("volumeK8s", "netpolicy"): - # Handle volumeK8s and netpolicy deletion - appinstance.state = "Deleted" - appinstance.deleted_on = timezone.now() - status = AppStatus(appinstance=appinstance) - status.status_type = "Deleted" - appinstance.save() - status.save() +@app.task +def delete_old_objects(): + """ + This function retrieves the old apps based on the given threshold, category, and model class. + It then iterates through the subclasses of BaseAppInstance and deletes the old apps + for both the "Develop" and "Manage Files" categories. -@shared_task -@transaction.atomic -def delete_and_deploy_resource(instance_pk, new_release_name): - appinstance = AppInstance.objects.select_for_update().get(pk=instance_pk) - - if appinstance: - delete_resource(appinstance.pk) - - parameters = appinstance.parameters - parameters["release"] = new_release_name - appinstance.parameters.update(parameters) - appinstance.save(update_fields=["parameters", "table_field"]) - - try: - rel_name_obj = ReleaseName.objects.get(name=new_release_name, project=appinstance.project, status="active") - rel_name_obj.status = "in-use" - rel_name_obj.app = appinstance - rel_name_obj.save() - except Exception: - logger.error("Error: Submitted release name not owned by project.", exc_info=True) - - deploy_resource(appinstance.pk) - - -@shared_task -@transaction.atomic -def deploy_resource(instance_pk, action="create"): - logger.info("TASK - DEPLOY RESOURCE...") - appinstance = AppInstance.objects.select_for_update().get(pk=instance_pk) - - results = controller.deploy(appinstance.parameters) - if type(results) is str: - results = json.loads(results) - stdout = results["status"] - stderr = results["reason"] - logger.info("Helm install failed") - helm_info = { - "success": False, - "info": {"stdout": stdout, "stderr": stderr}, - } - appinstance.info["helm"] = helm_info - appinstance.save() - else: - stdout, stderr = process_helm_result(results) - - if results.returncode == 0: - logger.info("Helm install succeeded") - - helm_info = { - "success": True, - "info": {"stdout": stdout, "stderr": stderr}, - } - else: - logger.error("Helm install failed") - helm_info = { - "success": False, - "info": {"stdout": stdout, "stderr": stderr}, - } - appinstance.info["helm"] = helm_info - appinstance.save() - if results.returncode != 0: - logger.info(appinstance.info["helm"]) - else: - post_create_hooks(appinstance) - return results + """ + def get_threshold(threshold): + return timezone.now() - timezone.timedelta(days=threshold) -@shared_task -@transaction.atomic -def delete_resource(pk): - appinstance = AppInstance.objects.select_for_update().get(pk=pk) - - if appinstance and appinstance.state != "Deleted": - # The instance does exist. - parameters = appinstance.parameters - # TODO: Check that the user has the permission required to delete it. - - # TODO: Fix for multicluster setup - # TODO: We are assuming this URI here, but we should allow - # for other forms. - # The instance should store information about this. - - # Invoke chart controller - results = controller.delete(parameters) - - if results.returncode == 0 or "release: not found" in results.stderr.decode("utf-8"): - status = AppStatus(appinstance=appinstance) - status.status_type = "Deleting..." - appinstance.state = "Deleting..." - status.save() - logger.info("CALLING POST DELETE HOOKS") - post_delete_hooks(appinstance) - else: - status = AppStatus(appinstance=appinstance) - status.status_type = "FailedToDelete" - status.save() - appinstance.state = "FailedToDelete" - appinstance.save(update_fields=["state"]) + # Handle deletion of apps in the "Develop" category + for orm_model in APP_REGISTRY.iter_orm_models(): + old_develop_apps = orm_model.objects.filter(created_on__lt=get_threshold(7), app__category__name="Develop") + for app_ in old_develop_apps: + delete_resource.delay(app_.pk) -@shared_task -@transaction.atomic -def delete_resource_permanently(appinstance): - parameters = appinstance.parameters + # Handle deletion of non persistent file managers + old_file_managers = FilemanagerInstance.objects.filter( + created_on__lt=timezone.now() - timezone.timedelta(days=1), persistent=False + ) + for app_ in old_file_managers: + delete_resource.delay(app_.pk) - # Invoke chart controller - results = controller.delete(parameters) - if not (results.returncode == 0 or "release: not found" in results.stderr.decode("utf-8")): - status = AppStatus(appinstance=appinstance) - status.status_type = "FailedToDelete" - status.save() - appinstance.state = "FailedToDelete" +def helm_install(release_name, chart, namespace="default", values_file=None, version=None): + """ + Run a Helm install command. - release_name(appinstance) + Args: + release_name (str): Name of the Helm release. + chart (str): Helm chart to install. + namespace (str): Kubernetes namespace to deploy to. + options (list, optional): Additional options for Helm command in list format. - appinstance.delete() + Returns: + tuple: Output message and any errors from the Helm command. + """ + # Base command + command = f"helm upgrade --force --install {release_name} {chart} --namespace {namespace}" + if values_file: + command += f" -f {values_file}" -@app.task -def get_resource_usage(): - timestamp = time.time() + # Append version if deploying via ghcr + if version: + command += f" --version {version} --repository-cache /app/charts/.cache/helm/repository" - args = ["kubectl", "get", "--raw", "/apis/metrics.k8s.io/v1beta1/pods"] - results = subprocess.run(args, capture_output=True) - - pods = [] + logger.debug(f"Running Helm command: {command}") + # Execute the command try: - res_json = json.loads(results.stdout.decode("utf-8")) - pods = res_json["items"] - except: # noqa E722 TODO: Add exception - pass - - resources = dict() + result = subprocess.run(command.split(" "), check=True, text=True, capture_output=True) + return result.stdout, None + except subprocess.CalledProcessError as e: + return e.stdout, e.stderr - args_pod = ["kubectl", "-n", f"{settings.NAMESPACE}" "get", "po", "-o", "json"] - results_pod = subprocess.run(args_pod, capture_output=True) - results_pod_json = json.loads(results_pod.stdout.decode("utf-8")) - try: - for pod in results_pod_json["items"]: - if ( - "metadata" in pod - and "labels" in pod["metadata"] - and "release" in pod["metadata"]["labels"] - and "project" in pod["metadata"]["labels"] - ): - pod_name = pod["metadata"]["name"] - resources[pod_name] = dict() - resources[pod_name]["labels"] = pod["metadata"]["labels"] - resources[pod_name]["cpu"] = 0.0 - resources[pod_name]["memory"] = 0.0 - resources[pod_name]["gpu"] = 0 - except: # noqa E722 TODO: Add exception - pass +@shared_task +def helm_delete(release_name, namespace="default"): + # Base command + command = f"helm uninstall {release_name} --namespace {namespace} --wait" + # Execute the command try: - for pod in pods: - podname = pod["metadata"]["name"] - if podname in resources: - containers = pod["containers"] - cpu = 0 - mem = 0 - for container in containers: - cpun = container["usage"]["cpu"] - memki = container["usage"]["memory"] - try: - cpu += int(cpun.replace("n", "")) / 1e6 - except: # noqa E722 TODO: Add exception - logger.error("Failed to parse CPU usage: %s", cpun) - if "Ki" in memki: - mem += int(memki.replace("Ki", "")) / 1000 - elif "Mi" in memki: - mem += int(memki.replace("Mi", "")) - elif "Gi" in memki: - mem += int(memki.replace("Mi", "")) * 1000 - - resources[podname]["cpu"] = cpu - resources[podname]["memory"] = mem - except: # noqa E722 TODO: Add exception - pass - # TODO minor refactor: remove unnecessary comments - # print(json.dumps(resources, indent=2)) - - for key in resources.keys(): - entry = resources[key] - # print(entry['labels']['release']) - try: - appinstance = AppInstance.objects.get(parameters__contains={"release": entry["labels"]["release"]}) - # print(timestamp) - # print(appinstance) - # print(entry) - datapoint = ResourceData( - appinstance=appinstance, - cpu=entry["cpu"], - mem=entry["memory"], - gpu=entry["gpu"], - time=timestamp, - ) - datapoint.save() - except: # noqa E722 TODO: Add exception - logger.error("Didn't find corresponding AppInstance: %s", key, exc_info=True) - - # print(timestamp) - # print(json.dumps(resources, indent=2)) - - -@app.task -def sync_mlflow_models(): - mlflow_apps = AppInstance.objects.filter(~Q(state="Deleted"), project__status="active", app__slug="mlflow") - for mlflow_app in mlflow_apps: - if mlflow_app.project is None or mlflow_app.project.mlflow is None: - continue - - url = "http://{}/{}".format( - mlflow_app.project.mlflow.host, - "api/2.0/mlflow/model-versions/search", - ) - res = False - try: - res = requests.get(url) - except Exception: - logger.error("Call to MLFlow Server failed.", exc_info=True) - - if res: - models = res.json() - logger.info(models) - if len(models) > 0: - for item in models["model_versions"]: - # print(item) - name = item["name"] - version = "v{}.0.0".format(item["version"]) - release = "major" - source = item["source"].replace("s3://", "").split("/") - run_id = source[2] - path = "/".join(source[1:]) - project = mlflow_app.project - uid = run_id - s3 = S3.objects.get(pk=mlflow_app.parameters["s3"]["pk"]) - model_found = True - try: - stackn_model = Model.objects.get(uid=uid) - except: # noqa E722 TODO: Add exception - model_found = False - if not model_found: - obj_type = ObjectType.objects.filter(slug="mlflow") - if obj_type.exists(): - model = Model( - version=version, - project=project, - name=name, - uid=uid, - release_type=release, - s3=s3, - bucket="mlflow", - path=path, - ) - model.save() - model.object_type.set(obj_type) - else: - raise EmptyResultSet - else: - if item["current_stage"] == "Archived" and stackn_model.status != "AR": - stackn_model.status = "AR" - stackn_model.save() - if item["current_stage"] != "Archived" and stackn_model.status == "AR": - stackn_model.status = "CR" - stackn_model.save() - else: - logger.warning("WARNING: Failed to fetch info from MLflow Server: %s", url) + result = subprocess.run(command.split(" "), check=True, text=True, capture_output=True) + return result.stdout, None + except subprocess.CalledProcessError as e: + return e.stdout, e.stderr -@app.task -def clean_resource_usage(): - curr_timestamp = time.time() - ResourceData.objects.filter(time__lte=curr_timestamp - 48 * 3600).delete() +@shared_task +@transaction.atomic +def deploy_resource(serialized_instance): + instance = deserialize(serialized_instance) + logger.info("Deploying resource for instance %s", instance) + values = instance.k8s_values + if "ghcr" in instance.chart: + version = instance.chart.split(":")[-1] + chart = "oci://" + instance.chart.split(":")[0] + # Save helm values file for internal reference + values_file = f"charts/values/{str(uuid.uuid4())}.yaml" + with open(values_file, "w") as f: + f.write(yaml.dump(values)) + output, error = helm_install(values["subdomain"], chart, values["namespace"], values_file, version) + success = not error -@app.task -def remove_deleted_app_instances(): - apps = AppInstance.objects.filter(state="Deleted") - logger.info("NUMBER OF APPS TO DELETE: %s", len(apps)) - for instance in apps: - try: - name = instance.name - logger.info("Deleting app instance: %s", name) - instance.delete() - logger.info("Deleted app instance: %s", name) - except Exception: - logger.error("Failed to delete app instances.", exc_info=True) + helm_info = {"success": success, "info": {"stdout": output, "stderr": error}} + instance.info = dict(helm=helm_info) + instance.app_status.status = "Created" if success else "Failed" -@app.task -def clear_table_field(): - all_apps = AppInstance.objects.all() - for instance in all_apps: - instance.table_field = "{}" - instance.save() + instance.app_status.save() + instance.save() - all_apps = Apps.objects.all() - for instance in all_apps: - instance.table_field = "{}" - instance.save() + subprocess.run(["rm", "-f", values_file]) -@app.task -def purge_tasks(): - """ - Remove tasks from queue to avoid overflow - """ - app.control.purge() +@shared_task +@transaction.atomic +def delete_resource(serialized_instance): + instance = deserialize(serialized_instance) + values = instance.k8s_values + output, error = helm_delete(values["subdomain"], values["namespace"]) + success = not error -@app.task -def delete_old_objects(): - """ - Deletes apps of category Develop (e.g., jupyter-lab, vscode etc) + if success: + if instance.app.slug in ("volumeK8s", "netpolicy"): + instance.app_status.status = "Deleted" + instance.deleted_on = datetime.now() + else: + instance.app_status.status = "Deleting..." + else: + instance.app_status.status = "FailedToDelete" - Setting the threshold to 7 days. If any app is older than this, it will be deleted. - The deleted resource will still exist in the database, but with status "Deleted" + helm_info = {"success": success, "info": {"stdout": output, "stderr": error}} - TODO: Make this a variable in settings.py and use the same number in templates - """ + instance.info = dict(helm=helm_info) + instance.app_status.save() + instance.save() - def get_old_apps(threshold, category): - threshold_time = timezone.now() - timezone.timedelta(days=threshold) - return AppInstance.objects.filter(created_on__lt=threshold_time, app__category__name=category) - old_develop_apps = get_old_apps(threshold=7, category="Develop") - old_minio_apps = get_old_apps(threshold=1, category="Manage Files") - for app_ in old_develop_apps: - delete_resource.delay(app_.pk) +def deserialize(serialized_instance): + # Check if the input is a dictionary + if not isinstance(serialized_instance, dict): + raise ValueError(f"The input must be a dictionary and not {type(serialized_instance)}") - for app_ in old_minio_apps: - delete_resource.delay(app_.pk) + try: + model = serialized_instance["model"] + pk = serialized_instance["pk"] + app_label, model_name = model.split(".") + + model_class = apps.get_model(app_label, model_name) + instance = model_class.objects.get(pk=pk) + + return instance + except (KeyError, ValueError) as e: + raise ValueError(f"Invalid serialized data format: {e}") + except ObjectDoesNotExist: + raise ValueError(f"No instance found for model {model} with pk {pk}") diff --git a/apps/tests/test_app_instance.py b/apps/tests/test_app_instance.py index 38568c7cc..ad52d0aa8 100644 --- a/apps/tests/test_app_instance.py +++ b/apps/tests/test_app_instance.py @@ -1,69 +1,102 @@ from django.contrib.auth import get_user_model +from django.db.models.signals import post_save from django.test import TestCase from projects.models import Project -from ..models import AppInstance, Apps +from ..models import ( + Apps, + AppStatus, + BaseAppInstance, + CustomAppInstance, + FilemanagerInstance, + JupyterInstance, + RStudioInstance, + ShinyInstance, + Subdomain, + TissuumapsInstance, + VSCodeInstance, +) User = get_user_model() +MODELS_LIST = [ + JupyterInstance, + RStudioInstance, + VSCodeInstance, + FilemanagerInstance, + ShinyInstance, + TissuumapsInstance, + CustomAppInstance, +] -class AppInstaceTestCase(TestCase): + +class AppInstanceTestCase(TestCase): def setUp(self): self.user = User.objects.create_user("foo1", "foo@test.com", "bar") def get_data(self, access): project = Project.objects.create_project(name="test-perm", owner=self.user, description="") - app = Apps.objects.create(name="FEDn Combiner", slug="combiner") - - app_instance = AppInstance.objects.create( - access=access, - owner=self.user, - name="test_app_instance_private", - app=app, - project=project, - ) - - return [project, app, app_instance] + app = Apps.objects.create(name="Serve App", slug="serve-app") + + app_instance_list = [] + for i, model_class in enumerate(MODELS_LIST): + subdomain = Subdomain.objects.create(subdomain=f"test_internal_{i}") + app_status = AppStatus.objects.create(status="Created") + + app_instance = model_class.objects.create( + access=access, + owner=self.user, + name="test_app_instance_private", + app=app, + project=project, + subdomain=subdomain, + app_status=app_status, + ) + app_instance_list.append(app_instance) + + return app_instance_list def test_permission_created_if_private(self): - project, app, app_instance = self.get_data("private") + app_instance_list = self.get_data("private") - result = self.user.has_perm("can_access_app", app_instance) - - self.assertTrue(result) + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertTrue(all(result_list)) def test_permission_do_note_created_if_project(self): - project, app, app_instance = self.get_data("project") - - result = self.user.has_perm("can_access_app", app_instance) + app_instance_list = self.get_data("project") - self.assertFalse(result) + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertFalse(any(result_list)) def test_permission_create_if_changed_to_private(self): - project, app, app_instance = self.get_data("project") + app_instance_list = self.get_data("project") - result = self.user.has_perm("can_access_app", app_instance) + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertFalse(any(result_list)) - self.assertFalse(result) + for app_instance in app_instance_list: + app_instance.access = "private" + app_instance.save() - app_instance.access = "private" - app_instance.save() - - result = self.user.has_perm("can_access_app", app_instance) - - self.assertTrue(result) + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertTrue(all(result_list)) def test_permission_remove_if_changed_from_project(self): - project, app, app_instance = self.get_data("private") - - result = self.user.has_perm("can_access_app", app_instance) - - self.assertTrue(result) + app_instance_list = self.get_data("private") - app_instance.access = "project" - app_instance.save() + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertTrue(all(result_list)) - result = self.user.has_perm("can_access_app", app_instance) + for app_instance in app_instance_list: + app_instance.access = "project" + app_instance.save() - self.assertFalse(result) + result_list = [self.user.has_perm("can_access_app", app_instance) for app_instance in app_instance_list] + print(result_list) + self.assertFalse(any(result_list)) diff --git a/apps/tests/test_app_instance_manager.py b/apps/tests/test_app_instance_manager.py index fbbcc5a84..199eaa1a1 100644 --- a/apps/tests/test_app_instance_manager.py +++ b/apps/tests/test_app_instance_manager.py @@ -4,10 +4,31 @@ from projects.models import Project -from ..models import AppInstance, Apps +from ..models import ( + Apps, + BaseAppInstance, + CustomAppInstance, + FilemanagerInstance, + JupyterInstance, + RStudioInstance, + ShinyInstance, + Subdomain, + TissuumapsInstance, + VSCodeInstance, +) User = get_user_model() +MODELS_LIST = [ + JupyterInstance, + RStudioInstance, + VSCodeInstance, + FilemanagerInstance, + ShinyInstance, + TissuumapsInstance, + CustomAppInstance, +] + class AppInstaceManagerTestCase(TestCase): def setUp(self): @@ -17,72 +38,17 @@ def setUp(self): ) app = Apps.objects.create(name="Persistent Volume", slug="volumeK8s") - app_instance = AppInstance.objects.create( - access="project", - owner=self.user, - name="test_app_instance_1", - app=app, - project=self.project, - ) - - app_instance_2 = AppInstance.objects.create( - access="project", - owner=self.user, - name="test_app_instance_2", - app=app, - project=self.project, - ) - - app_instance_3 = AppInstance.objects.create( - access="project", - owner=self.user, - name="test_app_instance_3", - app=app, - project=self.project, - ) - - app_instance.app_dependencies.set([app_instance_2, app_instance_3]) - - app_instance_4 = AppInstance.objects.create( - access="project", - owner=self.user, - name="test_app_instance_4", - app=app, - project=self.project, - ) - - app_instance_5 = AppInstance.objects.create( - access="project", - owner=self.user, - name="test_app_instance_5", - app=app, - project=self.project, - ) - - app_instance_4.app_dependencies.set([app_instance_5]) - - @override_settings(STUDIO_ACCESSMODE="ReadWriteOnce") - def test_get_available_app_dependencies_rw_once(self): - result = AppInstance.objects.get_available_app_dependencies( - user=self.user, project=self.project, app_name="Persistent Volume" - ) - - self.assertEqual(len(result), 2) - - @override_settings(STUDIO_ACCESSMODE="ReadWriteMany") - def test_get_available_app_dependencies_rw_many(self): - result = AppInstance.objects.get_available_app_dependencies( - user=self.user, project=self.project, app_name="Persistent Volume" - ) - - self.assertEqual(len(result), 5) - - def test_get_available_app_dependencies_setting_default(self): - result = AppInstance.objects.get_available_app_dependencies( - user=self.user, project=self.project, app_name="Persistent Volume" - ) - - self.assertEqual(len(result), 5) + self.instances = [] + for i, model_class in enumerate(MODELS_LIST): + subdomain = Subdomain.objects.create(subdomain=f"test_{i}") + instance = BaseAppInstance.objects.create( + owner=self.user, + name=f"test_app_instance_{i}", + app=app, + project=self.project, + subdomain=subdomain, + ) + self.instances.append(instance) # ---------- get_app_instances_of_project ---------- # @@ -93,66 +59,67 @@ def test_get_app_instances_of_project(self): app = Apps.objects.create(name="Combiner", slug="combiner") - app_instance = AppInstance.objects.create( - access="project", + subdomain = Subdomain.objects.create(subdomain="test_internal") + instance = BaseAppInstance.objects.create( owner=self.user, name="test_app_instance_internal", app=app, project=project, + subdomain=subdomain, ) - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project) - self.assertEqual(len(result), 5) + self.assertEqual(len(result), len(MODELS_LIST)) - app_instance.project = self.project - app_instance.save() + instance.project = self.project + instance.save() - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project) - self.assertEqual(len(result), 6) + self.assertEqual(len(result), len(MODELS_LIST) + 1) - app_instance.access = "private" - app_instance.save() + instance.access = "private" + instance.save() - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project) - self.assertEqual(len(result), 6) + self.assertEqual(len(result), len(MODELS_LIST) + 1) def test_get_app_instances_of_project_limit(self): - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project, limit=3) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project, limit=3) self.assertEqual(len(result), 3) - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project) - self.assertEqual(len(result), 5) + self.assertEqual(len(result), len(MODELS_LIST)) def test_get_app_instances_of_project_filter(self): app = Apps.objects.create(name="Combiner", slug="combiner") - - _ = AppInstance.objects.create( - access="project", + subdomain = Subdomain.objects.create(subdomain="test_internal") + _ = BaseAppInstance.objects.create( owner=self.user, name="test_app_instance_internal", app=app, project=self.project, + subdomain=subdomain, ) def filter_func(slug): return Q(app__slug=slug) - result = AppInstance.objects.get_app_instances_of_project( + result = BaseAppInstance.objects.get_app_instances_of_project( self.user, self.project, filter_func=filter_func("volumeK8s") ) - self.assertEqual(len(result), 5) + self.assertEqual(len(result), len(MODELS_LIST)) - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project) + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project) - self.assertEqual(len(result), 6) + self.assertEqual(len(result), len(MODELS_LIST) + 1) - result = AppInstance.objects.get_app_instances_of_project( + result = BaseAppInstance.objects.get_app_instances_of_project( self.user, self.project, filter_func=filter_func("non-existing-slug"), @@ -161,14 +128,14 @@ def filter_func(slug): self.assertEqual(len(result), 0) def test_get_app_instances_of_project_order_by(self): - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project, order_by="name") + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project, order_by="name") - self.assertEqual(len(result), 5) - self.assertEqual(result.first().name, "test_app_instance_1") - self.assertEqual(result.last().name, "test_app_instance_5") + self.assertEqual(len(result), len(MODELS_LIST)) + self.assertEqual(result.first().name, "test_app_instance_0") + self.assertEqual(result.last().name, f"test_app_instance_{len(MODELS_LIST)-1}") - result = AppInstance.objects.get_app_instances_of_project(self.user, self.project, order_by="-name") + result = BaseAppInstance.objects.get_app_instances_of_project(self.user, self.project, order_by="-name") - self.assertEqual(len(result), 5) - self.assertEqual(result.first().name, "test_app_instance_5") - self.assertEqual(result.last().name, "test_app_instance_1") + self.assertEqual(len(result), len(MODELS_LIST)) + self.assertEqual(result.first().name, f"test_app_instance_{len(MODELS_LIST)-1}") + self.assertEqual(result.last().name, "test_app_instance_0") diff --git a/apps/tests/test_app_settings_view.py b/apps/tests/test_app_settings_view.py index 73655e506..1fe673959 100644 --- a/apps/tests/test_app_settings_view.py +++ b/apps/tests/test_app_settings_view.py @@ -3,7 +3,7 @@ from projects.models import Project -from ..models import AppCategories, AppInstance, Apps +from ..models import AppCategories, Apps, AppStatus, CustomAppInstance, Subdomain User = get_user_model() @@ -15,39 +15,25 @@ def setUp(self) -> None: self.user = User.objects.create_user(test_user["username"], test_user["email"], test_user["password"]) self.category = AppCategories.objects.create(name="Network", priority=100, slug="network") self.app = Apps.objects.create( - name="Jupyter Lab", - slug="jupyter-lab", + name="My Custom App", + slug="customapp", user_can_edit=False, category=self.category, - settings={ - "apps": {"Persistent Volume": "many"}, - "flavor": "one", - "default_values": {"port": "80", "targetport": "8888"}, - "environment": { - "name": "from", - "title": "Image", - "quantity": "one", - "type": "match", - }, - "permissions": { - "public": {"value": "false", "option": "false"}, - "project": {"value": "true", "option": "true"}, - "private": {"value": "false", "option": "true"}, - "link": {"value": "false", "option": "true"}, - }, - "export-cli": "True", - }, ) self.project = Project.objects.create_project(name="test-perm", owner=self.user, description="") - self.app_instance = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + self.app_instance = CustomAppInstance.objects.create( access="public", owner=self.user, name="test_app_instance_public", app=self.app, project=self.project, - parameters={ + subdomain=subdomain, + app_status=app_status, + k8s_values={ "environment": {"pk": ""}, }, ) @@ -67,7 +53,7 @@ def test_user_can_edit_true(self): self.assertEqual(response.status_code, 302) - url = f"/projects/{self.project.slug}/" + f"apps/settings/{self.app_instance.id}" + url = f"/projects/{self.project.slug}/" + f"apps/settings/{self.app_instance.app.slug}/{self.app_instance.id}" response = c.get(url) @@ -84,7 +70,7 @@ def test_user_can_edit_false(self): self.app.user_can_edit = True self.app.save() - url = f"/projects/{self.project.slug}/" + f"apps/settings/{self.app_instance.id}" + url = f"/projects/{self.project.slug}/" + f"apps/settings/{self.app_instance.app.slug}/{self.app_instance.id}" response = c.get(url) diff --git a/apps/tests/test_app_view_forbidden.py b/apps/tests/test_app_view_forbidden.py index 07de63f67..954c8fc53 100644 --- a/apps/tests/test_app_view_forbidden.py +++ b/apps/tests/test_app_view_forbidden.py @@ -18,74 +18,6 @@ def setUp(self): user = User.objects.create_user("member", "bar@test.com", "bar") self.client.login(username="bar@test.com", password="bar") - def test_forbidden_apps_compute(self): - """ - Test non-project member not allowed to access /=compute - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:filtered", - kwargs={ - "project": project.slug, - "category": "compute", - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_serve(self): - """ - Test non-project member not allowed to access /=serve - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:filtered", - kwargs={ - "project": project.slug, - "category": "serve", - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_store(self): - """ - Test non-project member not allowed to access /=store - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:filtered", - kwargs={ - "project": project.slug, - "category": "store", - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_develop(self): - """ - Test non-project member not allowed to access /=develop - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:filtered", - kwargs={ - "project": project.slug, - "category": "develop", - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - def test_forbidden_apps_create(self): """ Test non-project member not allowed to access /create/=test @@ -105,13 +37,13 @@ def test_forbidden_apps_create(self): def test_forbidden_apps_logs(self): """ - Test non-project member not allowed to access /logs/=1 + Test non-project member not allowed to access /logs/=jupyter-lab/=1 """ project = Project.objects.get(name="test-perm") response = self.client.get( reverse( "apps:logs", - kwargs={"project": project.slug, "ai_id": "1"}, + kwargs={"project": project.slug, "app_slug": "jupyter-lab", "app_id": "1"}, ) ) self.assertTemplateUsed(response, "403.html") @@ -119,82 +51,11 @@ def test_forbidden_apps_logs(self): def test_forbidden_apps_settings(self): """ - Test non-project member not allowed to access /seetings/=1 - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:appsettings", - kwargs={"project": project.slug, "ai_id": "1"}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_settings_add_tag(self): - """ - Test non-project member not allowed to access - /settings/=1/add_tag - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:add_tag", - kwargs={"project": project.slug, "ai_id": "1"}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_settings_remove_tag(self): - """ - Test non-project member not allowed to access - /settings/=1/remove_tag - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:remove_tag", - kwargs={"project": project.slug, "ai_id": "1"}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_delete(self): - """ - Test non-project member not allowed to access - /delete/=compute/=1 + Test non-project member not allowed to access /settings/=jupyter-lab/=1 """ project = Project.objects.get(name="test-perm") response = self.client.get( - reverse( - "apps:delete", - kwargs={ - "project": project.slug, - "ai_id": "1", - "category": "compute", - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_apps_publish(self): - """ - Test non-project member not allowed to access - /publish/=compute/=1 - """ - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "apps:publish", - kwargs={ - "project": project.slug, - "ai_id": "1", - "category": "compute", - }, - ) + reverse("apps:appsettings", kwargs={"project": project.slug, "app_slug": "jupyter-lab", "app_id": "1"}) ) self.assertTemplateUsed(response, "403.html") self.assertEqual(response.status_code, 403) diff --git a/apps/tests/test_create_app_view.py b/apps/tests/test_create_app_view.py index 9c14c9188..899edbb20 100644 --- a/apps/tests/test_create_app_view.py +++ b/apps/tests/test_create_app_view.py @@ -5,7 +5,7 @@ from projects.models import Project -from ..models import AppInstance, Apps +from ..models import Apps, AppStatus, JupyterInstance, Subdomain User = get_user_model() @@ -18,24 +18,6 @@ def setUp(self) -> None: self.app = Apps.objects.create( name="Jupyter Lab", slug="jupyter-lab", - settings={ - "apps": {"Persistent Volume": "many"}, - "flavor": "one", - "default_values": {"port": "80", "targetport": "8888"}, - "environment": { - "name": "from", - "title": "Image", - "quantity": "one", - "type": "match", - }, - "permissions": { - "public": {"value": "false", "option": "false"}, - "project": {"value": "true", "option": "true"}, - "private": {"value": "false", "option": "true"}, - "link": {"value": "false", "option": "true"}, - }, - "export-cli": "True", - }, ) def get_data(self, user=None): @@ -163,12 +145,16 @@ def test_has_permission_project_level(self): self.assertEqual(response.status_code, 200) - _ = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + _ = JupyterInstance.objects.create( access="private", owner=self.user, name="test_app_instance_private", app=self.app, project=project, + subdomain=subdomain, + app_status=app_status, ) response = c.get(f"/projects/{project.slug}/apps/create/jupyter-lab") @@ -190,11 +176,11 @@ def test_permission_overrides_reached_app_limit(self): self.assertEqual(response.status_code, 403) - content_type = ContentType.objects.get_for_model(AppInstance) + content_type = ContentType.objects.get_for_model(JupyterInstance) project_permissions = Permission.objects.filter(content_type=content_type) add_permission = next( - (perm for perm in project_permissions if perm.codename == "add_appinstance"), + (perm for perm in project_permissions if perm.codename == "add_jupyterinstance"), None, ) @@ -230,12 +216,16 @@ def test_app_limit_is_per_project(self): self.assertEqual(response.status_code, 200) - _ = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + _ = JupyterInstance.objects.create( access="private", owner=self.user, name="test_app_instance_private", app=self.app, project=project, + subdomain=subdomain, + app_status=app_status, ) response = c.get(f"/projects/{project.slug}/apps/create/jupyter-lab") @@ -280,12 +270,16 @@ def test_app_limit_altered_for_project_v2(self): self.assertEqual(response.status_code, 200) - _ = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + _ = JupyterInstance.objects.create( access="private", owner=self.user, name="test_app_instance_private", app=self.app, project=project, + subdomain=subdomain, + app_status=app_status, ) response = c.get(f"/projects/{project.slug}/apps/create/jupyter-lab") diff --git a/apps/tests/test_delete_app_view.py b/apps/tests/test_delete_app_view.py index c80668c09..56e237bb9 100644 --- a/apps/tests/test_delete_app_view.py +++ b/apps/tests/test_delete_app_view.py @@ -5,7 +5,7 @@ from projects.models import Project -from ..models import AppCategories, AppInstance, Apps +from ..models import AppCategories, Apps, AppStatus, JupyterInstance, Subdomain User = get_user_model() @@ -21,33 +21,20 @@ def setUp(self) -> None: slug="jupyter-lab", user_can_delete=False, category=self.category, - settings={ - "apps": {"Persistent Volume": "many"}, - "flavor": "one", - "default_values": {"port": "80", "targetport": "8888"}, - "environment": { - "name": "from", - "title": "Image", - "quantity": "one", - "type": "match", - }, - "permissions": { - "public": {"value": "false", "option": "false"}, - "project": {"value": "true", "option": "true"}, - "private": {"value": "false", "option": "true"}, - }, - "export-cli": "True", - }, ) self.project = Project.objects.create_project(name="test-perm", owner=self.user, description="") - self.app_instance = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + self.app_instance = JupyterInstance.objects.create( access="public", owner=self.user, name="test_app_instance_public", app=self.app, project=self.project, + subdomain=subdomain, + app_status=app_status, ) def get_data(self, user=None): @@ -65,7 +52,7 @@ def test_user_can_delete_false(self): self.assertEqual(response.status_code, 302) - url = f"/projects/{self.project.slug}/apps/delete/" + f"{self.category.slug}/{self.app_instance.id}" + url = f"/projects/{self.project.slug}/apps/delete/" + f"{self.app_instance.app.slug}/{self.app_instance.id}" response = c.get(url) @@ -83,13 +70,13 @@ def test_user_can_delete_true(self): self.app.save() with patch("apps.tasks.delete_resource.delay") as mock_task: - url = f"/projects/{self.project.slug}/apps/delete/" + f"{self.category.slug}/{self.app_instance.id}" + url = f"/projects/{self.project.slug}/apps/delete/" + f"{self.app_instance.app.slug}/{self.app_instance.id}" response = c.get(url) self.assertEqual(response.status_code, 302) - self.app_instance = AppInstance.objects.get(name="test_app_instance_public") + self.app_instance = JupyterInstance.objects.get(name="test_app_instance_public") self.assertEqual("private", self.app_instance.access) diff --git a/apps/tests/test_forms.py b/apps/tests/test_forms.py new file mode 100644 index 000000000..d98490520 --- /dev/null +++ b/apps/tests/test_forms.py @@ -0,0 +1,192 @@ +from django.contrib.auth import get_user_model +from django.template import Context, Template +from django.test import TestCase + +from apps.forms import CustomAppForm +from apps.models import Apps, AppStatus, Subdomain, VolumeInstance +from projects.models import Flavor, Project + +User = get_user_model() + +test_user = {"username": "foo1", "email": "foo@test.com", "password": "bar"} + + +class BaseAppFormTest(TestCase): + def setUp(self): + self.user = User.objects.create_user(test_user["username"], test_user["email"], test_user["password"]) + self.project = Project.objects.create_project(name="test-perm", owner=self.user, description="") + self.app = Apps.objects.create(name="Custom App", slug="customapp") + self.volume = VolumeInstance.objects.create( + name="project-vol", + app=self.app, + owner=self.user, + project=self.project, + size=1, + subdomain=Subdomain.objects.create(subdomain="subdomain", project=self.project), + app_status=AppStatus.objects.create(status="Created"), + ) + self.flavor = Flavor.objects.create(name="flavor", project=self.project) + + +class CustomAppFormTest(BaseAppFormTest): + def setUp(self): + super().setUp() + self.valid_data = { + "name": "Valid Name", + "description": "A valid description", + "subdomain": "valid-subdomain", + "volume": self.volume, + "path": "/home/user", + "flavor": self.flavor, + "access": "public", + "source_code_url": "http://example.com", + "note_on_linkonly_privacy": None, + "port": 8000, + "image": "ghcr.io/scilifelabdatacentre/image:tag", + "tags": ["tag1", "tag2", "tag3"], + } + + def test_form_valid_data(self): + form = CustomAppForm(self.valid_data, project_pk=self.project.pk) + self.assertTrue(form.is_valid()) + + def test_form_missing_data(self): + invalid_data = self.valid_data.copy() + invalid_data.pop("name") + invalid_data.pop("port") + invalid_data.pop("image") + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("name", form.errors) + self.assertIn("port", form.errors) + self.assertIn("image", form.errors) + + def test_invalid_path(self): + invalid_data = self.valid_data.copy() + invalid_data["path"] = "/var" + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("Path must start with", str(form.errors)) + + def test_valid_path_if_set_in_admin_panel(self): + valid_data = self.valid_data.copy() + form = CustomAppForm(valid_data, project_pk=self.project.pk) + self.assertTrue(form.is_valid()) + + # Simulate a path set in the admin panel + instance = form.save(commit=False) + instance.project = self.project + instance.owner = self.user + instance.app_status = AppStatus.objects.create(status="Created") + instance.app = self.app + + # Fetch subdomain and set + subdomain_name = form.cleaned_data.get("subdomain") + subdomain, _ = Subdomain.objects.get_or_create(subdomain=subdomain_name, project=self.project) + instance.subdomain = subdomain + + # Change the path to something that the form would not allow + instance.path = "/var" + instance.save() + + # Open form in "edit mode" by sending instance as argument + form = CustomAppForm(valid_data, project_pk=self.project.pk, instance=instance) + + # This should now work, since we set the path in the "admin panel" + self.assertTrue(form.is_valid()) + + def test_volume_no_path(self): + invalid_data = self.valid_data.copy() + invalid_data.pop("path") + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("Path is required when volume is selected.", str(form.errors)) + + def test_path_no_volume(self): + invalid_data = self.valid_data.copy() + invalid_data.pop("volume") + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("Warning, you have provided a path, but not selected a volume.", str(form.errors)) + + def test_invalid_subdomain(self): + invalid_data = self.valid_data.copy() + invalid_data["subdomain"] = "-some_invalid_subdomain!" + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("Subdomain must be 3-53 characters long", str(form.errors)) + + def test_source_url_enforced_when_public(self): + invalid_data = self.valid_data.copy() + invalid_data["source_code_url"] = "" + + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + + def test_link_only_note_enforced_when_link(self): + invalid_data = self.valid_data.copy() + invalid_data["access"] = "link" + + # Test no note + form = CustomAppForm(invalid_data, project_pk=self.project.pk) + self.assertFalse(form.is_valid()) + self.assertIn("Please, provide a reason for making the app accessible only via a link.", str(form.errors)) + + # Now add a note + valid_data = self.valid_data.copy() + valid_data["access"] = "link" + valid_data["note_on_linkonly_privacy"] = "A reason" + form = CustomAppForm(valid_data, project_pk=self.project.pk) + self.assertTrue(form.is_valid()) + + +class CustomAppFormRenderingTest(BaseAppFormTest): + def setUp(self): + super().setUp() + self.valid_data = { + "name": "Valid Name", + "description": "A valid description", + "subdomain": "valid-subdomain", + "volume": self.volume, + "path": "/home/user", + "flavor": self.flavor, + "access": "public", + "source_code_url": "http://example.com", + "note_on_linkonly_privacy": None, + "port": 8000, + "image": "ghcr.io/scilifelabdatacentre/image:tag", + "tags": ["tag1", "tag2", "tag3"], + } + + def test_form_rendering(self): + valid_data = self.valid_data.copy() + form = CustomAppForm(valid_data, project_pk=self.project.pk) + + template = Template("{% load crispy_forms_tags %}{% crispy form %}") + context = Context({"form": form}) + rendered_form = template.render(context) + for key, value in valid_data.items(): + if key == "tags": + value = "".join(tag for tag in key) + if key == "volume": + value = self.volume.name + if key == "flavor": + value = self.flavor.name + if key == "port": + value = str(value) + if value is None: + continue + + self.assertIn(value, rendered_form) + self.assertIn(f'name="{key}"', rendered_form) + self.assertIn(f'id="id_{key}"', rendered_form) + + self.assertIn('value="project"', rendered_form) + self.assertIn('value="private"', rendered_form) + self.assertIn('value="link"', rendered_form) + self.assertIn('value="public"', rendered_form) diff --git a/apps/tests/test_generate_form.py b/apps/tests/test_generate_form.py deleted file mode 100644 index eadd19bcf..000000000 --- a/apps/tests/test_generate_form.py +++ /dev/null @@ -1,305 +0,0 @@ -from copy import deepcopy - -from django.contrib.auth import get_user_model -from django.test import TestCase, override_settings - -from projects.models import Environment, Project - -from ..generate_form import get_form_environments, get_form_primitives -from ..models import AppInstance, Apps - -User = get_user_model() - - -class GenerateFormTestCase(TestCase): - def setUp(self) -> None: - self.app_settings_pvc = { - "volume": { - "size": { - "type": "select", - "title": "Size", - "default": "1Gi", - "user_can_edit": False, - "items": [ - {"name": "1Gi", "value": "1Gi"}, - {"name": "2Gi", "value": "2Gi"}, - {"name": "5Gi", "value": "5Gi"}, - ], - }, - "accessModes": { - "type": "string", - "title": "AccessModes", - "default": "ReadWriteMany", - }, - "storageClass": { - "type": "string", - "title": "StorageClass", - "default": "", - }, - }, - "permissions": { - "public": {"value": "false", "option": "false"}, - "private": {"value": "false", "option": "true"}, - "project": {"value": "true", "option": "true"}, - }, - "default_values": {"port": "port", "targetport": "targetport"}, - "environment": { - "name": "from", - "type": "match", - "title": "Image", - "quantity": "one", - }, - } - self.user = User.objects.create_user("foo1", "foo@test.com", "bar") - - self.project = Project.objects.create_project(name="test-perm-generate_form", owner=self.user, description="") - self.app = Apps.objects.create(name="Persistent Volume", slug="volumeK8s") - super().setUp() - - # primatives - - @override_settings(DISABLED_APP_INSTANCE_FIELDS=[]) - def test_get_form_primitives_should_return_complete(self): - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_primitives(app_settings, None) - - result_items = result["volume"] - result_keys = result_items.keys() - expected_items = self.app_settings_pvc["volume"].items() - - result_length = len(result_items) - expected_length = len(expected_items) + 1 - - self.assertEqual(result_length, expected_length) - - for key, val in expected_items: - is_in_keys = key in result_keys - - self.assertTrue(is_in_keys) - - is_string = isinstance(val, str) - - if is_string: - continue - - result_item = result_items[key] - - for _key, _val in val.items(): - result_val = result_item[_key] - self.assertEqual(result_val, _val) - - @override_settings(DISABLED_APP_INSTANCE_FIELDS=["accessModes", "storageClass"]) - def test_get_form_primitives_should_remove_two(self): - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_primitives(app_settings, None) - - result_items = result["volume"] - result_keys = result_items.keys() - - result_length = len(result_items) - expected_length = 2 - - self.assertEqual(result_length, expected_length) - - has_expected_keys = "meta_title" in result_keys and "size" in result_keys - - self.assertTrue(has_expected_keys) - - @override_settings( - DISABLED_APP_INSTANCE_FIELDS=[ - "accessModes", - "storageClass", - "madeUpValue", - ] - ) - def test_get_form_primitives_field_not_in_model(self): - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_primitives(app_settings, None) - - result_items = result["volume"] - result_keys = result_items.keys() - - result_length = len(result_items) - expected_length = 2 - - self.assertEqual(result_length, expected_length) - - has_expected_keys = "meta_title" in result_keys and "size" in result_keys - - self.assertTrue(has_expected_keys) - - @override_settings(DISABLED_APP_INSTANCE_FIELDS=[]) - def test_get_form_primitives_should_set_default(self): - app_settings = deepcopy(self.app_settings_pvc) - - app_instance = AppInstance( - name="My app", - access="private", - app=self.app, - project=self.project, - parameters={ - "volume": { - "size": "5Gi", - "accessModes": "ReadWriteMany", - "storageClass": "microk8s-hostpath", - }, - }, - owner=self.user, - ) - app_instance.save() - - result = get_form_primitives(app_settings, app_instance) - - result_items = result["volume"] - - result_size = result_items["size"]["default"] - result_size_user_can_edit = result_items["size"]["user_can_edit"] - result_access_modes = result_items["accessModes"]["default"] - result_storage_class = result_items["storageClass"]["default"] - - self.assertEqual(result_size, "5Gi") - self.assertFalse(result_size_user_can_edit) - self.assertEqual(result_access_modes, "ReadWriteMany") - self.assertEqual(result_storage_class, "microk8s-hostpath") - - app_instance.parameters = {} - - app_instance.save() - - app_settings_default = deepcopy(self.app_settings_pvc) - - app_instance = AppInstance.objects.get(name="My app") - - result = get_form_primitives(app_settings_default, app_instance) - - result_items = result["volume"] - - result_size = result_items["size"]["default"] - result_access_modes = result_items["accessModes"]["default"] - result_storage_class = result_items["storageClass"]["default"] - - self.assertEqual(result_size, "1Gi") - self.assertEqual(result_access_modes, "ReadWriteMany") - self.assertEqual(result_storage_class, "") - - # environments - - def test_get_form_environments_single(self): - environment = Environment( - app=self.app, - project=self.project, - name="test", - slug="test", - repository="test-repo", - image="test-image", - ) - - environment.save() - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_environments(app_settings, self.project, self.app, None) - - dep_environment, environments = result - - self.assertEqual(dep_environment, True) - - objs = environments["objs"] - - self.assertEqual(len(objs), 1) - - result_item = objs[0] - - self.assertEqual(result_item.name, "test") - self.assertEqual(result_item.slug, "test") - - def test_get_form_environments_with_public(self): - environment = Environment( - app=self.app, - project=self.project, - name="test1", - slug="test1", - repository="test1-repo", - image="test1-image", - ) - - environment.save() - - environment2 = Environment( - app=self.app, - name="test2", - slug="test2", - repository="test2-repo", - image="test2-image", - public=True, - ) - - environment2.save() - - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_environments(app_settings, self.project, self.app, None) - - dep_environment, environments = result - - self.assertEqual(dep_environment, True) - - objs = environments["objs"] - - self.assertEqual(len(objs), 2) - - number_of_public = 0 - - for obj in objs: - self.assertIn(obj.name, ["test1", "test2"]) - self.assertIn(obj.slug, ["test1", "test2"]) - - if obj.public: - number_of_public += 1 - - self.assertEqual(number_of_public, 1) - - def test_get_form_environments_with_public_and_other_projects(self): - project = Project.objects.create_project(name="test-perm-generate_form2", owner=self.user, description="") - - environment = Environment( - app=self.app, - project=project, - name="test1", - slug="test1", - repository="test1-repo", - image="test1-image", - ) - - environment.save() - - environment2 = Environment( - app=self.app, - name="test2", - slug="test2", - repository="test2-repo", - image="test2-image", - public=True, - ) - - environment2.save() - - app_settings = deepcopy(self.app_settings_pvc) - - result = get_form_environments(app_settings, self.project, self.app, None) - - dep_environment, environments = result - - self.assertEqual(dep_environment, True) - - objs = environments["objs"] - - self.assertEqual(len(objs), 1) - - result_item = objs[0] - - self.assertEqual(result_item.name, "test2") - self.assertEqual(result_item.slug, "test2") - self.assertTrue(result_item.public) diff --git a/apps/tests/test_get_status_view.py b/apps/tests/test_get_status_view.py index 4130724bb..ae5c06802 100644 --- a/apps/tests/test_get_status_view.py +++ b/apps/tests/test_get_status_view.py @@ -3,7 +3,7 @@ from projects.models import Project -from ..models import AppCategories, AppInstance, Apps +from ..models import AppCategories, Apps, AppStatus, JupyterInstance, Subdomain User = get_user_model() @@ -23,15 +23,16 @@ def setUp(self) -> None: self.project = Project.objects.create_project(name="test-perm-get_status", owner=self.user, description="") - self.app_instance = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + self.app_instance = JupyterInstance.objects.create( access="public", owner=self.user, name="test_app_instance_public", app=self.app, project=self.project, - parameters={ - "environment": {"pk": ""}, - }, + subdomain=subdomain, + app_status=app_status, ) def test_user_has_access(self): diff --git a/apps/tests/test_subdomain_candidate.py b/apps/tests/test_subdomain_candidate.py new file mode 100644 index 000000000..b7b8687a5 --- /dev/null +++ b/apps/tests/test_subdomain_candidate.py @@ -0,0 +1,62 @@ +import pytest +from django.core.exceptions import ValidationError + +from ..types_.subdomain import SubdomainCandidateName + + +@pytest.mark.django_db +def test_is_available_for_available_subdomains(): + """Tests for available subdomains""" + candidate = SubdomainCandidateName("test-name-is-unique") + assert candidate.is_available() + + candidate = SubdomainCandidateName("test-name-999-is-unique") + assert candidate.is_available() + + +def test_is_available_for_unavailable_subdomains(): + """Tests for unavailable subdomains""" + + # Test using the reserved word serve + candidate = SubdomainCandidateName("serve") + assert not candidate.is_available() + + +@pytest.mark.parametrize("name", [("test-with-hyphens"), ("0n9"), ("z-a"), ("01234567890")]) +def test_is_valid_for_valid_names(name): + """Tests for valid subdomain names""" + candidate = SubdomainCandidateName(name) + assert candidate.is_valid() + + +@pytest.mark.parametrize("name", [("a"), ("Test-Uppercase"), ("-test-starthyphen"), ("test-endhyphen-")]) +def test_is_valid_for_invalid_names(name): + """Tests for invalid subdomain names""" + candidate = SubdomainCandidateName(name) + assert not candidate.is_valid() + + +def test_validate_subdomain(): + """ + Tests validity of subdomain names. + Expects a raised ValidationError exception for invalid names. + """ + + # Test with valid subdomain + candidate = SubdomainCandidateName("test-subdomain") + candidate.validate_subdomain() + + # Test with invalid subdomain (too short) + candidate = SubdomainCandidateName("a") + with pytest.raises(ValidationError): + candidate.validate_subdomain() + + # Test with invalid subdomain (contains uppercase) + candidate = SubdomainCandidateName("Test-Subdomain") + with pytest.raises(ValidationError): + candidate.validate_subdomain() + + # Test with invalid subdomain (starts with hyphen) + candidate = SubdomainCandidateName("-test-subdomain") + with pytest.raises(ValidationError): + candidate.validate_subdomain() diff --git a/apps/tests/test_update_status_handler.py b/apps/tests/test_update_status_handler.py index eaee26c4a..3bd81fed2 100644 --- a/apps/tests/test_update_status_handler.py +++ b/apps/tests/test_update_status_handler.py @@ -11,7 +11,7 @@ from projects.models import Project from ..helpers import HandleUpdateStatusResponseCode, handle_update_status_request -from ..models import AppCategories, AppInstance, Apps, AppStatus +from ..models import AppCategories, Apps, AppStatus, JupyterInstance, Subdomain utc = pytz.UTC @@ -49,77 +49,79 @@ def setUp(self) -> None: self.project = Project.objects.create_project(name="test-perm-get_status", owner=self.user, description="") - self.app_instance = AppInstance.objects.create( - access="public", + subdomain = Subdomain.objects.create(subdomain=self.ACTUAL_RELEASE_NAME) + self.app_instance = JupyterInstance.objects.create( + access="private", owner=self.user, - name="test_app_instance_public", + name="test_app_instance_private", app=self.app, project=self.project, - parameters={ + subdomain=subdomain, + k8s_values={ "environment": {"pk": ""}, "release": self.ACTUAL_RELEASE_NAME, }, - state=self.INITIAL_STATUS, ) + print(f"######## {self.INITIAL_EVENT_TS}") def setUpCreateAppStatus(self): - self.status_object = AppStatus(appinstance=self.app_instance) - self.status_object.status_type = self.INITIAL_STATUS - self.status_object.save() - # Must re-save with the desired timeUpdate the app instance object - self.status_object.time = self.INITIAL_EVENT_TS - self.status_object.save(update_fields=["time"]) + app_status = AppStatus.objects.create(status=self.INITIAL_STATUS) + app_status.time = self.INITIAL_EVENT_TS + app_status.save() + self.app_instance.app_status = app_status + self.app_instance.save(update_fields=["app_status"]) + print(f"######## {app_status.time}") def test_handle_old_event_time_should_ignore_update(self): self.setUpCreateAppStatus() - older_ts = self.INITIAL_EVENT_TS - timedelta(seconds=1) + older_ts = self.app_instance.app_status.time - timedelta(seconds=1) actual = handle_update_status_request(self.ACTUAL_RELEASE_NAME, "NewStatus", older_ts) assert actual == HandleUpdateStatusResponseCode.NO_ACTION # Fetch the app instance and status objects and verify values - actual_app_instance = AppInstance.objects.filter( - parameters__contains={"release": self.ACTUAL_RELEASE_NAME} + actual_app_instance = JupyterInstance.objects.filter( + k8s_values__contains={"release": self.ACTUAL_RELEASE_NAME} ).last() - assert actual_app_instance.state == self.INITIAL_STATUS - actual_appstatus = actual_app_instance.status.latest() - assert actual_appstatus.status_type == self.INITIAL_STATUS + assert actual_app_instance.app_status.status == self.INITIAL_STATUS + actual_appstatus = actual_app_instance.app_status + assert actual_appstatus.status == self.INITIAL_STATUS assert actual_appstatus.time == self.INITIAL_EVENT_TS def test_handle_same_status_newer_time_should_update_time(self): self.setUpCreateAppStatus() - newer_ts = self.INITIAL_EVENT_TS + timedelta(seconds=1) + newer_ts = self.app_instance.app_status.time + timedelta(seconds=1) actual = handle_update_status_request(self.ACTUAL_RELEASE_NAME, self.INITIAL_STATUS, newer_ts) assert actual == HandleUpdateStatusResponseCode.UPDATED_TIME_OF_STATUS # Fetch the app instance and status objects and verify values - actual_app_instance = AppInstance.objects.filter( - parameters__contains={"release": self.ACTUAL_RELEASE_NAME} + actual_app_instance = JupyterInstance.objects.filter( + k8s_values__contains={"release": self.ACTUAL_RELEASE_NAME} ).last() - assert actual_app_instance.state == self.INITIAL_STATUS - actual_appstatus = actual_app_instance.status.latest() - assert actual_appstatus.status_type == self.INITIAL_STATUS + assert actual_app_instance.app_status.status == self.INITIAL_STATUS + actual_appstatus = actual_app_instance.app_status + assert actual_appstatus.status == self.INITIAL_STATUS assert actual_appstatus.time == newer_ts def test_handle_different_status_newer_time_should_update_status(self): self.setUpCreateAppStatus() - newer_ts = self.INITIAL_EVENT_TS + timedelta(seconds=1) + newer_ts = self.app_instance.app_status.time + timedelta(seconds=1) new_status = self.INITIAL_STATUS + "-test01" actual = handle_update_status_request(self.ACTUAL_RELEASE_NAME, new_status, newer_ts) assert actual == HandleUpdateStatusResponseCode.UPDATED_STATUS # Fetch the app instance and status objects and verify values - actual_app_instance = AppInstance.objects.filter( - parameters__contains={"release": self.ACTUAL_RELEASE_NAME} + actual_app_instance = JupyterInstance.objects.filter( + k8s_values__contains={"release": self.ACTUAL_RELEASE_NAME} ).last() - assert actual_app_instance.state == new_status - actual_appstatus = actual_app_instance.status.latest() - assert actual_appstatus.status_type == new_status + assert actual_app_instance.app_status.status == new_status + actual_appstatus = actual_app_instance.app_status + assert actual_appstatus.status == new_status assert actual_appstatus.time == newer_ts def test_handle_missing_app_status_should_create_and_update_status(self): @@ -130,13 +132,13 @@ def test_handle_missing_app_status_should_create_and_update_status(self): assert actual == HandleUpdateStatusResponseCode.CREATED_FIRST_STATUS # Fetch the app instance and status objects and verify values - actual_app_instance = AppInstance.objects.filter( - parameters__contains={"release": self.ACTUAL_RELEASE_NAME} + actual_app_instance = JupyterInstance.objects.filter( + k8s_values__contains={"release": self.ACTUAL_RELEASE_NAME} ).last() - assert actual_app_instance.state == new_status - actual_appstatus = actual_app_instance.status.latest() - assert actual_appstatus.status_type == new_status + assert actual_app_instance.app_status.status == new_status + actual_appstatus = actual_app_instance.app_status + assert actual_appstatus.status == new_status assert actual_appstatus.time == newer_ts def test_handle_long_status_text_should_trim_status(self): @@ -149,16 +151,18 @@ def test_handle_long_status_text_should_trim_status(self): assert actual == HandleUpdateStatusResponseCode.CREATED_FIRST_STATUS # Fetch the app instance and status objects and verify values - actual_app_instance = AppInstance.objects.filter( - parameters__contains={"release": self.ACTUAL_RELEASE_NAME} + actual_app_instance = JupyterInstance.objects.filter( + k8s_values__contains={"release": self.ACTUAL_RELEASE_NAME} ).last() - assert actual_app_instance.state == expected_status_text - actual_appstatus = actual_app_instance.status.latest() - assert actual_appstatus.status_type == expected_status_text + assert actual_app_instance.app_status.status == expected_status_text + actual_appstatus = actual_app_instance.app_status + assert actual_appstatus.status == expected_status_text assert actual_appstatus.time == newer_ts +''' +#TODO: THIS TEST NEEDS TO BE UPDATED TO ADHERE TO NEW LOGIC @pytest.mark.skip( reason="This test requires a modification to the handle_update_status_request function to add a delay parameter." ) @@ -269,3 +273,4 @@ def submit_request_new_status(self): new_status = "StatusB" actual = handle_update_status_request(self.ACTUAL_RELEASE_NAME, new_status, newer_ts) return "submit_request_new_status", actual +''' diff --git a/collections_module/__init__.py b/apps/types_/__init__.py similarity index 100% rename from collections_module/__init__.py rename to apps/types_/__init__.py diff --git a/apps/types_/app_registry.py b/apps/types_/app_registry.py new file mode 100644 index 000000000..07555e178 --- /dev/null +++ b/apps/types_/app_registry.py @@ -0,0 +1,35 @@ +from apps.types_.app_types import ModelFormTuple, OptionalModelFormTuple + + +class AppRegistry: + def __init__(self): + self._apps: dict[str, ModelFormTuple] = {} + + def register(self, app_slug: str, app: ModelFormTuple): + self._apps[app_slug] = app + + def __getitem__(self, app_slug: str) -> OptionalModelFormTuple: + return self._apps.get(app_slug, (None, None)) + + def get(self, app_slug: str) -> OptionalModelFormTuple: + return self[app_slug] + + def get_apps(self): + return self._apps + + def get_orm_model(self, app_slug: str): + return self[app_slug][0] + + def get_form_class(self, app_slug: str): + return self[app_slug][1] + + def iter_orm_models(self): + for app in self._apps.values(): + yield app.Model + + def iter_forms(self): + for app in self._apps.values(): + yield app.Form + + def __contains__(self, item): + return item in self._apps diff --git a/apps/types_/app_types.py b/apps/types_/app_types.py new file mode 100644 index 000000000..186f98dad --- /dev/null +++ b/apps/types_/app_types.py @@ -0,0 +1,14 @@ +from typing import NamedTuple, Type, Union + +from apps.forms import BaseForm +from apps.models import BaseAppInstance + + +class ModelFormTuple(NamedTuple): + Model: Type[BaseAppInstance] + Form: Type[BaseForm] + + +NoneTuple = tuple[None, None] + +OptionalModelFormTuple = Union[ModelFormTuple, NoneTuple] diff --git a/apps/types_/subdomain.py b/apps/types_/subdomain.py new file mode 100644 index 000000000..52b61ebfc --- /dev/null +++ b/apps/types_/subdomain.py @@ -0,0 +1,49 @@ +from django.core.exceptions import ValidationError +from django.core.validators import RegexValidator + +from apps.models import Subdomain + + +class SubdomainCandidateName: + """ + A candidate subdomain name that may or may not be available or allowed. + """ + + __name = None + + def __init__(self, name: str): + self.__name = name + + def is_available(self) -> bool: + """Determines if the candidate name is available in Serve.""" + if self.__name == "serve": + # Reserved words + return False + elif Subdomain.objects.filter(subdomain=self.__name).exists(): + return False + else: + return True + + def is_valid(self) -> bool: + """Determines if the candidate name is valid.""" + try: + self.validate_subdomain() + return True + except ValidationError: + return False + + def validate_subdomain(self): + """ + Validates a subdomain text. + The RegexValidator will raise a ValidationError if the input does not match the regular expression. + It is up to the caller to handle the raised exception if desired. + """ + + # Check if the subdomain adheres to helm rules + regex_validator = RegexValidator( + regex=r"^(?!-)[a-z0-9-]{3,53}(?", views.FilteredView.as_view(), name="filtered"), - path("create/", CreateView.as_view(), name="create"), - path("create//create_releasename", views.create_releasename, name="create_releasename"), - path("serve//", CreateServeView.as_view(), name="serve"), - path("logs/", views.GetLogsView.as_view(), name="logs"), - path("settings/", AppSettingsView.as_view(), name="appsettings"), - path("settings//add_tag", views.add_tag, name="add_tag"), - path("settings//remove_tag", views.remove_tag, name="remove_tag"), - path("delete//", views.delete, name="delete"), - path("publish//", views.publish, name="publish"), - path("unpublish//", views.unpublish, name="unpublish"), + path("logs", views.GetLogs.as_view(), name="get_logs"), + path("logs//", views.GetLogs.as_view(), name="logs"), + path("create/", views.CreateApp.as_view(), name="create"), + path("settings//", views.CreateApp.as_view(), name="appsettings"), + path("delete//", views.delete, name="delete"), ] diff --git a/apps/views.py b/apps/views.py index d822eb441..098a3b277 100644 --- a/apps/views.py +++ b/apps/views.py @@ -1,33 +1,25 @@ -import re from datetime import datetime import requests -from django.apps import apps from django.conf import settings from django.contrib.auth import get_user_model -from django.db.models import Q +from django.core.exceptions import PermissionDenied +from django.db import transaction from django.http import HttpResponseForbidden, JsonResponse from django.shortcuts import HttpResponseRedirect, render, reverse from django.utils.decorators import method_decorator from django.views import View from guardian.decorators import permission_required_or_403 -from projects.models import Flavor -from studio.settings import DOMAIN +from projects.models import Project from studio.utils import get_logger -from .generate_form import generate_form -from .helpers import can_access_app_instances, create_app_instance, handle_permissions -from .models import AppCategories, AppInstance, Apps -from .serialize import serialize_app -from .tasks import delete_and_deploy_resource, delete_resource, deploy_resource +from .app_registry import APP_REGISTRY +from .helpers import create_instance_from_form +from .tasks import delete_resource logger = get_logger(__name__) - -Project = apps.get_model(app_label=settings.PROJECTS_MODEL) -ReleaseName = apps.get_model(app_label=settings.RELEASENAME_MODEL) - User = get_user_model() @@ -37,122 +29,102 @@ def get_status_defs(): return status_success, status_warning -# Create your views here. -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def index(request, user, project): - category = "store" - template = "index_apps.html" - - cat_obj = AppCategories.objects.get(slug=category) - apps = Apps.objects.filter(category=cat_obj) - project = Project.objects.get(slug=project) - appinstances = AppInstance.objects.filter( - Q(owner=request.user) | Q(permission__projects__slug=project.slug) | Q(permission__public=True), - app__category=cat_obj, - ) - - return render(request, template, locals()) - - @method_decorator( permission_required_or_403("can_view_project", (Project, "slug", "project")), name="dispatch", ) -class GetLogsView(View): - def get(self, request, project, ai_id): - template = "apps/logs.html" - app = AppInstance.objects.get(pk=ai_id) - project = Project.objects.get(slug=project) - return render(request, template, locals()) +class GetLogs(View): + template = "apps/logs.html" - def post(self, request, project): - body = request.POST.get("app", "") - container = request.POST.get("container", "") - app = AppInstance.objects.get(pk=body) - project = Project.objects.get(slug=project) - app_settings = app.app.settings - logs = [] - # Looks for logs in app settings. TODO: this logs entry is not used. Remove or change this later. - if "logs" in app_settings: - try: - url = settings.LOKI_SVC + "/loki/api/v1/query_range" - app_params = app.parameters - if app.app.slug == "shinyproxyapp": - log_query = '{release="' + app_params["release"] + '",container="' + "serve" + '"}' - else: - log_query = '{release="' + app_params["release"] + '",container="' + container + '"}' - logger.info(log_query) - query = { - "query": log_query, - "limit": 500, - "since": "24h", - } - res = requests.get(url, params=query) - res_json = res.json()["data"]["result"] - # TODO: change timestamp logic. Timestamps are different in prod and dev - for item in res_json: - for log_line in reversed(item["values"]): - # separate timestamp and log - separated_log = log_line[1].split(None, 1) - # improve timestamp formatting for table - filtered_log = separated_log[0][:-4] if settings.DEBUG else separated_log[0][:-10] - formatted_time = datetime.strptime(filtered_log, "%Y-%m-%dT%H:%M:%S.%f") - separated_log[0] = datetime.strftime(formatted_time, "%Y-%m-%d, %H:%M:%S") - logs.append(separated_log) - - except Exception as e: - logger.error(str(e), exc_info=True) - - return JsonResponse({"data": logs}) - - -@method_decorator( - permission_required_or_403("can_view_project", (Project, "slug", "project")), - name="dispatch", -) -class FilteredView(View): - template_name = "apps/new.html" - - def get(self, request, user, project, category): - project = Project.objects.get(slug=project) + def get_instance(self, app_slug, app_id, post=False): + model_class = APP_REGISTRY.get_orm_model(app_slug) + if model_class: + return model_class.objects.get(pk=app_id) + else: + message = f"Could not find model for slug {app_slug}" + if post: + return JsonResponse({"error": message}, status=404) + else: + logger.error(message) + raise PermissionDenied() + + def get_project(self, project_slug, post=False): + try: + project = Project.objects.get(slug=project_slug) + return project + except Project.DoesNotExist: + message = "error: Project not found" + if post: + return JsonResponse({"error": message}, status=404) + else: + logger.error(message) + raise PermissionDenied() - def filter_func(): - filter = AppInstance.objects.get_app_instances_of_project_filter( - user=request.user, project=project, deleted_time_delta=5 - ) + def get(self, request, project, app_slug, app_id): + project = self.get_project(project) + instance = self.get_instance(app_slug, app_id) - filter &= Q(app__category__slug=category) + context = {"instance": instance, "project": project} + return render(request, self.template, context) - return filter + def post(self, request, project, app_slug, app_id): + # Validate project and instance existence + project = self.get_project(project, post=True) + instance = self.get_instance(app_slug, app_id, post=True) - app_instances_of_category = AppInstance.objects.filter(filter_func()).order_by("-created_on") + # container name is often same as subdomain name + container = instance.subdomain.subdomain - app_ids = [obj.id for obj in app_instances_of_category] + if not getattr(instance, "logs_enabled", False): + return JsonResponse({"error": "Logs not enabled for this instance"}, status=403) - apps_of_category = ( - Apps.objects.filter(category=category, user_can_create=True).order_by("slug", "-revision").distinct("slug") - ) + if not settings.LOKI_SVC: + return JsonResponse({"error": "LOKI_SVC not set"}, status=403) - def category2title(cat): - if cat == "compute": - return "Notebooks" - else: - return cat.capitalize() + logs = [] + try: + url = settings.LOKI_SVC + "/loki/api/v1/query_range" + container = "serve" if instance.app.slug == "shinyproxyapp" else container + log_query = f'{{release="{instance.subdomain.subdomain}",container="{container}"}}' + logger.info(f"Log query: {log_query}") + + query_params = { + "query": log_query, + "limit": 500, + "since": "24h", + } - context = { - "apps": apps_of_category, - "appinstances": app_instances_of_category, - "app_ids": app_ids, - "project": project, - "category": category, - "title": category2title(category), - } + res = requests.get(url, params=query_params) + res.raise_for_status() # Raise an HTTPError for bad responses (4xx and 5xx) + + res_json = res.json().get("data", {}).get("result", []) + + for item in res_json: + for log_line in reversed(item["values"]): + # Separate timestamp and log message + timestamp, log_message = log_line[0], log_line[1] + if len(log_message) < 2: + continue # Skip log lines that do not have a message + + # Parse and format the timestamp + try: + formatted_time = datetime.fromtimestamp(int(timestamp) / 1e9).strftime("%Y-%m-%d %H:%M:%S") + logs.append([formatted_time, log_message]) + except ValueError as ve: + logger.warning(f"Timestamp parsing failed: {ve}") + continue + + except requests.RequestException as e: + logger.error(f"HTTP request failed: {e}", exc_info=True) + return JsonResponse({"error": "Failed to retrieve logs from Loki"}, status=500) + except KeyError as e: + logger.error(f"Unexpected response format: {e}", exc_info=True) + return JsonResponse({"error": "Unexpected response format from Loki"}, status=500) + except Exception as e: + logger.error(f"An unexpected error occurred: {e}", exc_info=True) + return JsonResponse({"error": "An unexpected error occurred"}, status=500) - return render( - request=request, - context=context, - template_name=self.template_name, - ) + return JsonResponse({"data": logs}) @method_decorator( @@ -169,524 +141,136 @@ def post(self, request, project): arr = body.split(",") status_success, status_warning = get_status_defs() - app_instances = AppInstance.objects.filter(pk__in=arr) + for orm_model in APP_REGISTRY.iter_orm_models(): + instances = orm_model.objects.filter(pk__in=arr) - for instance in app_instances: - try: - status = instance.status.latest().status_type - except: # noqa E722 TODO: Add exception - status = instance.state + for instance in instances: + status_object = instance.app_status + if status_object: + status = status_object.status + else: + status = None - status_group = ( - "success" if status in status_success else "warning" if status in status_warning else "danger" - ) + status_group = ( + "success" if status in status_success else "warning" if status in status_warning else "danger" + ) - obj = { - "status": status, - "statusGroup": status_group, - } + obj = { + "status": status, + "statusGroup": status_group, + } - result[f"{instance.pk}"] = obj + result[f"{instance.app.slug}-{instance.pk}"] = obj return JsonResponse(result) return JsonResponse(result) -@method_decorator( - permission_required_or_403("can_view_project", (Project, "slug", "project")), - name="dispatch", -) -class AppSettingsView(View): - def get_shared_data(self, project_slug, ai_id): - project = Project.objects.get(slug=project_slug) - appinstance = AppInstance.objects.get(pk=ai_id) - - return [project, appinstance] - - def get(self, request, project, ai_id): - project, appinstance = self.get_shared_data(project, ai_id) - all_tags = AppInstance.tags.tag_model.objects.all() - template = "apps/update.html" - - existing_app_name = appinstance.name - existing_app_description = appinstance.description - existing_app_release_name = appinstance.parameters.get("release", None) - existing_userid = None - existing_path = None - existing_source_code_url = appinstance.source_code_url - existing_proxyheartbeatrate = None - existing_proxyheartbeattimeout = None - existing_proxycontainerwaittime = None - - # Settings for custom app and shinyproxy - if "appconfig" in appinstance.parameters: - appconfig = appinstance.parameters["appconfig"] - existing_userid = appconfig.get("userid", None) - if "path" in appconfig: - # check if app created by admin user then don't show path change option to normal user - created_by_admin = appinstance.parameters.get("created_by_admin") is True - existing_path = appconfig["path"] - - if not created_by_admin: - existing_path = existing_path.replace("/home/", "", 1) - existing_proxyheartbeatrate = appconfig.get("proxyheartbeatrate", None) - existing_proxyheartbeattimeout = appconfig.get("proxyheartbeattimeout", None) - existing_proxycontainerwaittime = appconfig.get("proxycontainerwaittime", None) - - app = appinstance.app - do_display_description_field = app.category.name is not None and app.category.name.lower() == "serve" - - if not app.user_can_edit: - return HttpResponseForbidden() - - app_settings = appinstance.app.settings - form = generate_form(app_settings, project, app, request.user, appinstance) - - # This handles the volume cases. If the app is mounted, then that volume should be pre-selected and vice-versa. - # Note that this assumes only ONE volume per app. - current_volumes = appinstance.parameters.get("apps", {}).get("volumeK8s", {}).keys() - current_volume = AppInstance.objects.filter(project=project, name__in=current_volumes).first() - - def filter_func(): - name = current_volume.name if current_volume else None - return Q(app__name="Persistent Volume") & ~Q(state="Deleted") & ~Q(name=name) - - available_volumes = AppInstance.objects.get_app_instances_of_project( - user=request.user, - project=project, - filter_func=filter_func(), - ) - - show_permissions = request.user.id == appinstance.owner.id or request.user.is_superuser - - context = { - "app": app, - "do_display_description_field": do_display_description_field, - "form": form, - "current_volume": current_volume, - "available_volumes": available_volumes, - "appinstance": appinstance, - "project": project, - "domain": DOMAIN, - "all_tags": all_tags, - "show_permissions": show_permissions, - "existing_app_name": existing_app_name, - "existing_path": existing_path, - "existing_app_description": existing_app_description, - "existing_app_release_name": existing_app_release_name, - "existing_userid": existing_userid, - "existing_source_code_url": existing_source_code_url, - "existing_proxyheartbeatrate": existing_proxyheartbeatrate, - "existing_proxyheartbeattimeout": existing_proxyheartbeattimeout, - "existing_proxycontainerwaittime": existing_proxycontainerwaittime, - } - - return render(request, template, context) - - def post(self, request, project, ai_id): - project, appinstance = self.get_shared_data(project, ai_id) - - app = appinstance.app - app_settings = app.settings - body = request.POST.copy() - - if not app.user_can_edit: - return HttpResponseForbidden() - - self.update_app_instance(request, project, appinstance, app_settings, body) - - return HttpResponseRedirect( - reverse( - "projects:details", - kwargs={ - "project_slug": str(project.slug), - }, - ) - ) - - def update_app_instance(self, request, project, appinstance, app_settings, body): - if not body.get("permission", None): - body.update({"permission": appinstance.access}) - current_release_name = appinstance.parameters["release"] - parameters, app_deps, model_deps = serialize_app(body, project, app_settings, request.user.username) - - authorized = can_access_app_instances(app_deps, request.user, project) - - if not authorized: - raise Exception("Not authorized to use specified app dependency") - - access = handle_permissions(parameters, project) - - flavor_id = request.POST.get("flavor") - appinstance.flavor = Flavor.objects.get(pk=flavor_id, project=project) - - appinstance.name = request.POST.get("app_name") - appinstance.description = request.POST.get("app_description") - appinstance.note_on_linkonly_privacy = body.get("link_privacy_type_note", "") - if "appconfig" in appinstance.parameters: - created_by_admin = False # default created by admin - userid = "1000" # default userid - if "path" in appinstance.parameters["appconfig"]: - # check if app created by admin user then don't show path change option to normal user - if "created_by_admin" in appinstance.parameters: - if appinstance.parameters["created_by_admin"] is True: - created_by_admin = True - existing_path = appinstance.parameters["appconfig"]["path"] - if "userid" in appinstance.parameters["appconfig"]: - userid = appinstance.parameters["appconfig"]["userid"] - appinstance.parameters.update(parameters) - appinstance.access = access - appinstance.app_dependencies.set(app_deps) - appinstance.model_dependencies.set(model_deps) - appinstance.source_code_url = request.POST.get("source_code_url") - if "appconfig" in appinstance.parameters and appinstance.app.slug == "customapp": - # remove trailing / in all cases - if appinstance.parameters["appconfig"]["path"] != "/": - appinstance.parameters["appconfig"]["path"] = appinstance.parameters["appconfig"]["path"].rstrip("/") - appinstance.parameters["created_by_admin"] = created_by_admin - # if app is created by admin but admin user is not updating it dont change path. - if created_by_admin: - if not request.user.is_superuser: - appinstance.parameters["appconfig"]["userid"] = userid - appinstance.parameters["appconfig"]["path"] = existing_path - else: - appinstance.parameters["appconfig"]["path"] = "/home/" + appinstance.parameters["appconfig"]["path"] - if not request.user.is_superuser: - appinstance.parameters["appconfig"]["userid"] = userid - - appinstance.save( - update_fields=[ - "flavor", - "name", - "description", - "parameters", - "access", - "note_on_linkonly_privacy", - "source_code_url", - ] - ) - self.update_resource(request, appinstance, current_release_name) - - def update_resource(self, request, appinstance, current_release_name): - domain = appinstance.parameters["global"]["domain"] - # if subdomain is set as --generated--, then use appname - new_release_name = request.POST.get("app_release_name") - if not new_release_name: - new_release_name = appinstance.parameters["appname"] - - new_url = f"https://{new_release_name}.{domain}" - appinstance.table_field.update({"url": new_url}) - - if new_release_name and current_release_name != new_release_name: - # This handles the case where a user creates a new subdomain, we must update the helm release aswell - delete_and_deploy_resource.delay(appinstance.pk, new_release_name) - else: - deploy_resource.delay(appinstance.pk, "update") - - appinstance.save() - - @permission_required_or_403("can_view_project", (Project, "slug", "project")) -def create_releasename(request, project, app_slug): - pattern = re.compile("^[a-z0-9][a-z0-9-]+[a-z0-9]$") - available = "invalid" - system_subdomains = ["keycloak", "grafana", "prometheus", "studio"] - if pattern.match(request.POST.get("rn")): - available = "false" - count_rn = ReleaseName.objects.filter(name=request.POST.get("rn")).count() - if count_rn == 0 and request.POST.get("rn") not in system_subdomains: - available = "true" - release = ReleaseName() - release.name = request.POST.get("rn") - release.status = "active" - release.project = Project.objects.get(slug=project) - release.save() - logger.info("RELEASE_NAME: %s %s", request.POST.get("rn"), count_rn) - return JsonResponse( - { - "available": available, - "rn": request.POST.get("rn"), - } - ) +def delete(request, project, app_slug, app_id): + model_class = APP_REGISTRY.get_orm_model(app_slug) + logger.info(f"Deleting app type {model_class} with id {app_id}") + if model_class is None: + raise PermissionDenied() -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def add_tag(request, project, ai_id): - appinstance = AppInstance.objects.get(pk=ai_id) - if request.method == "POST": - new_tags = request.POST.get("tag", "") - for new_tag in new_tags.split(","): - logger.info("New Tag: %s", new_tag) - appinstance.tags.add(new_tag.strip().lower().replace('"', "")) - appinstance.save() + instance = model_class.objects.get(pk=app_id) if app_id else None - return HttpResponseRedirect( - reverse( - "apps:appsettings", - kwargs={"project": project, "ai_id": ai_id}, - ) - ) + if instance is None: + raise PermissionDenied() + + if not instance.app.user_can_delete: + return HttpResponseForbidden() + serialized_instance = instance.serialize() -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def remove_tag(request, project, ai_id): - appinstance = AppInstance.objects.get(pk=ai_id) - if request.method == "POST": - logger.info(request.POST) - new_tag = request.POST.get("tag", "") - logger.info("Remove Tag: %s", new_tag) - appinstance.tags.remove(new_tag) - appinstance.save() + delete_resource.delay(serialized_instance) + # fix: in case appinstance is public swich to private + instance.access = "private" + instance.save() return HttpResponseRedirect( reverse( - "apps:appsettings", - kwargs={"project": project, "ai_id": ai_id}, + "projects:details", + kwargs={ + "project_slug": str(project), + }, ) ) -@method_decorator( - permission_required_or_403( - "can_view_project", - (Project, "slug", "project"), - ), - name="dispatch", -) -class CreateServeView(View): - def get_shared_data(self, project_slug, app_slug): - project = Project.objects.get(slug=project_slug) - app = Apps.objects.filter(slug=app_slug).order_by("-revision")[0] - app_settings = app.settings - - return [project, app, app_settings] - - def get(self, request, user, project, app_slug, version): - template = "apps/create.html" - project, app, app_settings = self.get_shared_data(project, app_slug) - domain = DOMAIN - user = request.user - if "from" in request.GET: - from_page = request.GET.get("from") - else: - from_page = "filtered" - - do_display_description_field = app.category is not None and app.category.name.lower() == "serve" - - form = generate_form(app_settings, project, app, user, []) - - for model in form["models"]: - if model.version == version: - model.selected = "selected" - - return render(request, template, locals()) - - @method_decorator( permission_required_or_403("can_view_project", (Project, "slug", "project")), name="dispatch", ) -class CreateView(View): - def get_shared_data(self, project_slug, app_slug): +class CreateApp(View): + template_name = "apps/create_view.html" + + def get(self, request, project, app_slug, app_id=None): + # TODO This is a bit confusing. project is actually project_slug. So it would be better to rename it + # Look in studio/urls.py There is . It's being passed from here there + # But need to make sure, that that's the only place where it's being passed + project_slug = project project = Project.objects.get(slug=project_slug) - app = Apps.objects.filter(slug=app_slug).order_by("-revision")[0] - app_settings = app.settings - - return [project, app, app_settings] - - def get(self, request, project, app_slug, data=[], wait=False, call=False): - template = "apps/create.html" - project, app, app_settings = self.get_shared_data(project, app_slug) - - domain = DOMAIN - if not call: - user = request.user - if "from" in request.GET: - from_page = request.GET.get("from") - else: - from_page = "filtered" - else: - from_page = "" - user = User.objects.get(username=user) - user_can_create = AppInstance.objects.user_can_create(user, project, app_slug) + form = self.get_form(request, project, app_slug, app_id) - if not user_can_create: - return HttpResponseForbidden() + if form is None or not getattr(form, "is_valid", False): + raise PermissionDenied() - do_display_description_field = app.category is not None and app.category.name.lower() == "serve" - - form = generate_form(app_settings, project, app, user, []) - - return render(request, template, locals()) - - def post(self, request, project, app_slug, data=[], wait=False): - project, app, app_settings = self.get_shared_data(project, app_slug) - data = request.POST - user = request.user - - user_can_create = AppInstance.objects.user_can_create(user, project, app_slug) + return render( + request, self.template_name, {"form": form, "project": project, "app_id": app_id, "app_slug": app_slug} + ) - if not user_can_create: - return HttpResponseForbidden() + @transaction.atomic + def post(self, request, project, app_slug, app_id=None): + # App id is used when updataing an instance - # Nikita Churikov @ nikita.churikov@scilifelab.uu.se on 25.01.2024 - # TODO: This is questionable but I won't touch it for now - # 1. We should not be throwing just a generic Exception - # 2. Couldn't we add this to the check above? - if not app.user_can_create: - raise Exception("User not allowed to create app") + # TODO Same as in get method + project_slug = project + project = Project.objects.get(slug=project_slug) - successful, project_slug, app_category_slug = create_app_instance(user, project, app, app_settings, data, wait) + form = self.get_form(request, project, app_slug, app_id) + if form is None: + raise PermissionDenied() - if not successful: - return HttpResponseRedirect( - reverse( - "projects:details", - kwargs={ - "project_slug": str(project.slug), - }, - ) - ) + if not form.is_valid(): + return render(request, self.template_name, {"form": form}) - if "from" in request.GET: - from_page = request.GET.get("from") - if from_page == "overview": - return HttpResponseRedirect( - reverse( - "projects:details", - kwargs={ - "project_slug": str(project_slug), - }, - ) - ) + # Otherwise we can create the instance + create_instance_from_form(form, project, app_slug, app_id) return HttpResponseRedirect( reverse( - "apps:filtered", + "projects:details", kwargs={ - "project": str(project_slug), - "category": app_category_slug, + "project_slug": str(project_slug), }, ) ) + def get_form(self, request, project, app_slug, app_id): + model_class, form_class = APP_REGISTRY.get(app_slug) -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def publish(request, user, project, category, ai_id): - try: - app = AppInstance.objects.get(pk=ai_id) - app.access = "public" - - if app.parameters["permissions"] is not None: - app.parameters["permissions"] = { - "public": True, - "project": False, - "private": False, - "link": False, - } + logger.info(f"Creating app type {model_class}") + if not model_class or not form_class: + logger.error("Could not fetch model or form") + return None - app.save() - except Exception as err: - logger.error(err, exc_info=True) + # Check if user is allowed + user_can_edit = False + user_can_create = False - return HttpResponseRedirect( - reverse( - "apps:filtered", - kwargs={ - "user": request.user, - "project": str(project), - "category": category, - }, - ) - ) - - -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def unpublish(request, user, project, category, ai_id): - try: - app = AppInstance.objects.get(pk=ai_id) - app.access = "project" - - if app.parameters["permissions"] is not None: - app.parameters["permissions"] = { - "public": False, - "project": True, - "private": False, - } - - app.save() - except Exception as err: - logger.error(err, exc_info=True) - - return HttpResponseRedirect( - reverse( - "apps:filtered", - kwargs={ - "user": request.user, - "project": str(project), - "category": category, - }, - ) - ) - - -@permission_required_or_403("can_view_project", (Project, "slug", "project")) -def delete(request, project, category, ai_id): - if "from" in request.GET: - from_page = request.GET.get("from") - else: - from_page = "filtered" - - app_instance = AppInstance.objects.get(pk=ai_id) - - if not app_instance.app.user_can_delete: - return HttpResponseForbidden() - - delete_resource.delay(ai_id) - # fix: in case appinstance is public swich to private - app_instance.access = "private" - app_instance.save() - - if "from" in request.GET: - from_page = request.GET.get("from") - if from_page == "overview": - return HttpResponseRedirect( - reverse( - "projects:details", - kwargs={ - "project_slug": str(project), - }, - ) - ) - elif from_page == "filtered": - return HttpResponseRedirect( - reverse( - "apps:filtered", - kwargs={ - "project": str(project), - "category": category, - }, - ) - ) + if app_id: + user_can_edit = model_class.objects.user_can_edit(request.user, project, app_slug) + instance = model_class.objects.get(pk=app_id) else: - return HttpResponseRedirect( - reverse( - "apps:filtered", - kwargs={ - "project": str(project), - "category": category, - }, - ) - ) + user_can_create = model_class.objects.user_can_create(request.user, project, app_slug) + instance = None - return HttpResponseRedirect( - reverse( - "apps:filtered", - kwargs={ - "project": str(project), - "category": category, - }, - ) - ) + if user_can_edit or user_can_create: + return form_class(request.POST or None, project_pk=project.pk, instance=instance) + # Maybe this makes typing hard. + else: + return None diff --git a/collections_module/admin.py b/collections_module/admin.py deleted file mode 100644 index 70ef7aa98..000000000 --- a/collections_module/admin.py +++ /dev/null @@ -1,15 +0,0 @@ -from django.contrib import admin - -from .models import Collection - - -class CollectionAdmin(admin.ModelAdmin): - readonly_fields = ["connected_apps"] - - def connected_apps(self, obj): - apps = obj.app_instances.all() - app_list = ", ".join([app.name for app in apps]) - return app_list or "No apps connected" - - -admin.site.register(Collection, CollectionAdmin) diff --git a/collections_module/apps.py b/collections_module/apps.py deleted file mode 100644 index beb230524..000000000 --- a/collections_module/apps.py +++ /dev/null @@ -1,6 +0,0 @@ -from django.apps import AppConfig - - -class CollectionsModuleConfig(AppConfig): - default_auto_field = "django.db.models.BigAutoField" - name = "collections_module" diff --git a/collections_module/migrations/0001_initial.py b/collections_module/migrations/0001_initial.py deleted file mode 100644 index a56c4ae28..000000000 --- a/collections_module/migrations/0001_initial.py +++ /dev/null @@ -1,37 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-20 15:02 - -import django.db.models.deletion -from django.conf import settings -from django.db import migrations, models - - -class Migration(migrations.Migration): - initial = True - - dependencies = [ - migrations.swappable_dependency(settings.AUTH_USER_MODEL), - ] - - operations = [ - migrations.CreateModel( - name="Collection", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("created_on", models.DateTimeField(auto_now_add=True)), - ("name", models.CharField(default="", max_length=200, unique=True)), - ("description", models.TextField(blank=True, default="", max_length=1000)), - ("website", models.URLField(blank=True)), - ("logo", models.ImageField(blank=True, null=True, upload_to="collections/logos/")), - ("slug", models.SlugField(blank=True, unique=True)), - ( - "maintainer", - models.ForeignKey( - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="collection_maintainer", - to=settings.AUTH_USER_MODEL, - ), - ), - ], - ), - ] diff --git a/collections_module/migrations/0002_collection_zenodo_community_id.py b/collections_module/migrations/0002_collection_zenodo_community_id.py deleted file mode 100644 index bb4589585..000000000 --- a/collections_module/migrations/0002_collection_zenodo_community_id.py +++ /dev/null @@ -1,17 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-28 17:17 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("collections_module", "0001_initial"), - ] - - operations = [ - migrations.AddField( - model_name="collection", - name="zenodo_community_id", - field=models.CharField(blank=True, max_length=200, null=True), - ), - ] diff --git a/collections_module/models.py b/collections_module/models.py deleted file mode 100644 index 6f3a6d2a1..000000000 --- a/collections_module/models.py +++ /dev/null @@ -1,28 +0,0 @@ -from django.contrib.auth import get_user_model -from django.db import models -from django.utils.text import slugify - - -class Collection(models.Model): - created_on = models.DateTimeField(auto_now_add=True) - maintainer = models.ForeignKey( - get_user_model(), - on_delete=models.CASCADE, - related_name="collection_maintainer", - null=True, - ) - name = models.CharField(max_length=200, unique=True, default="") - description = models.TextField(blank=True, default="", max_length=1000) - website = models.URLField(max_length=200, blank=True) - logo = models.ImageField(upload_to="collections/logos/", null=True, blank=True) - slug = models.SlugField(unique=True, blank=True) - zenodo_community_id = models.CharField(max_length=200, null=True, blank=True) - # repositories source would be another field - - def __str__(self): - return self.name - - def save(self, *args, **kwargs): - if not self.slug: - self.slug = slugify(self.name) - super(Collection, self).save(*args, **kwargs) diff --git a/collections_module/tests.py b/collections_module/tests.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/collections_module/urls.py b/collections_module/urls.py deleted file mode 100644 index a00fb3e0d..000000000 --- a/collections_module/urls.py +++ /dev/null @@ -1,10 +0,0 @@ -from django.urls import path - -from . import views - -app_name = "collections_module" - -urlpatterns = [ - path("collections/", views.index, name="index"), - path("collections//", views.collection, name="collection"), -] diff --git a/collections_module/views.py b/collections_module/views.py deleted file mode 100644 index d44b66649..000000000 --- a/collections_module/views.py +++ /dev/null @@ -1,36 +0,0 @@ -from django.apps import apps -from django.conf import settings -from django.db.models import Q -from django.shortcuts import get_object_or_404, render - -from portal.views import get_public_apps - -from .models import Collection - -PublishedModel = apps.get_model(app_label=settings.PUBLISHEDMODEL_MODEL) - - -def index(request): - template = "collections/index.html" - - collection_objects = Collection.objects.all().order_by("-created_on") - - context = {"collection_objects": collection_objects} - - return render(request, template, context=context) - - -def collection(request, slug, id=0): - template = "collections/collection.html" - - collection = get_object_or_404(Collection, slug=slug) - collection_published_apps, request = get_public_apps(request, id=id, collection=slug) - collection_published_models = PublishedModel.objects.all().filter(collections__slug=slug) - - context = { - "collection": collection, - "collection_published_apps": collection_published_apps, - "collection_published_models": collection_published_models, - } - - return render(request, template, context=context) diff --git a/common/management/commands/create_locust_apps.py b/common/management/commands/create_locust_apps.py index d324e2a3f..917448098 100644 --- a/common/management/commands/create_locust_apps.py +++ b/common/management/commands/create_locust_apps.py @@ -1,7 +1,8 @@ from django.contrib.auth import get_user_model from django.core.management.base import BaseCommand, CommandError -from apps.helpers import create_app_instance +# TODO: revisit: +from apps.helpers import create_app_instance # type:ignore[attr-defined] from apps.models import Apps from projects.models import Flavor, Project @@ -33,7 +34,7 @@ def handle(self, *args, **options): successful, project_slug, app_category_slug = create_app_instance( user, project, app, app.settings, data=data - ) + ) # TODO: revisit. type:ignore[attr-defined] else: self.stdout.write(self.style.WARNING(f"Flavor or app not found for user: {user.email}")) else: diff --git a/common/management/commands/install_fixtures.py b/common/management/commands/install_fixtures.py index 5931722d4..bba665c94 100644 --- a/common/management/commands/install_fixtures.py +++ b/common/management/commands/install_fixtures.py @@ -12,13 +12,13 @@ class Command(BaseCommand): def handle(self, *args, **kwargs): # Define all fixtures here files = [ - "projects_templates.json", - "intervals_fixtures.json", - "periodic_tasks_fixtures.json", "appcats_fixtures.json", "apps_fixtures.json", + "intervals_fixtures.json", + "periodic_tasks_fixtures.json", "objecttype_fixtures.json", "groups_fixtures.json", + "projects_templates.json", ] fixture_files = ["fixtures/" + file for file in files] diff --git a/common/migrations/0001_initial.py b/common/migrations/0001_initial.py index 05cd958db..b96ecccac 100644 --- a/common/migrations/0001_initial.py +++ b/common/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 +# Generated by Django 5.0.2 on 2024-05-27 07:28 import django.db.models.deletion from django.conf import settings @@ -13,12 +13,41 @@ class Migration(migrations.Migration): ] operations = [ + migrations.CreateModel( + name="FixtureVersion", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("filename", models.CharField(max_length=255, unique=True)), + ("hash", models.CharField(max_length=64)), + ], + ), + migrations.CreateModel( + name="MaintenanceMode", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("login_and_signup_disabled", models.BooleanField(default=False)), + ("message_in_header", models.TextField(blank=True, max_length=1000)), + ("message_in_footer", models.TextField(blank=True, max_length=1000)), + ], + ), + migrations.CreateModel( + name="EmailVerificationTable", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("token", models.CharField(max_length=100)), + ( + "user", + models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL), + ), + ], + ), migrations.CreateModel( name="UserProfile", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), ("affiliation", models.CharField(blank=True, max_length=100)), ("department", models.CharField(blank=True, max_length=100)), + ("deleted_on", models.DateTimeField(blank=True, null=True)), ("why_account_needed", models.TextField(blank=True, max_length=1000)), ("is_approved", models.BooleanField(default=False)), ("note", models.TextField(blank=True, max_length=1000)), diff --git a/common/migrations/0002_emailverificationtable.py b/common/migrations/0002_emailverificationtable.py deleted file mode 100644 index 5cbf987f0..000000000 --- a/common/migrations/0002_emailverificationtable.py +++ /dev/null @@ -1,26 +0,0 @@ -# Generated by Django 4.2.5 on 2023-11-24 13:27 - -import django.db.models.deletion -from django.conf import settings -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - migrations.swappable_dependency(settings.AUTH_USER_MODEL), - ("common", "0001_initial"), - ] - - operations = [ - migrations.CreateModel( - name="EmailVerificationTable", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("token", models.CharField(max_length=100)), - ( - "user", - models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL), - ), - ], - ), - ] diff --git a/common/migrations/0003_fixtureversion.py b/common/migrations/0003_fixtureversion.py deleted file mode 100644 index 33ac5e93e..000000000 --- a/common/migrations/0003_fixtureversion.py +++ /dev/null @@ -1,20 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-07 09:31 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("common", "0002_emailverificationtable"), - ] - - operations = [ - migrations.CreateModel( - name="FixtureVersion", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("filename", models.CharField(max_length=255, unique=True)), - ("hash", models.CharField(max_length=64)), - ], - ), - ] diff --git a/common/migrations/0004_userprofile_deleted_on.py b/common/migrations/0004_userprofile_deleted_on.py deleted file mode 100644 index a8ab363ab..000000000 --- a/common/migrations/0004_userprofile_deleted_on.py +++ /dev/null @@ -1,17 +0,0 @@ -# Generated by Django 4.2.7 on 2024-04-03 06:20 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("common", "0003_fixtureversion"), - ] - - operations = [ - migrations.AddField( - model_name="userprofile", - name="deleted_on", - field=models.DateTimeField(blank=True, null=True), - ), - ] diff --git a/common/migrations/0005_maintenancemode.py b/common/migrations/0005_maintenancemode.py deleted file mode 100644 index a8bc56551..000000000 --- a/common/migrations/0005_maintenancemode.py +++ /dev/null @@ -1,21 +0,0 @@ -# Generated by Django 4.2.7 on 2024-04-15 07:48 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("common", "0004_userprofile_deleted_on"), - ] - - operations = [ - migrations.CreateModel( - name="MaintenanceMode", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("login_and_signup_disabled", models.BooleanField(default=False)), - ("message_in_header", models.TextField(blank=True, max_length=1000)), - ("message_in_footer", models.TextField(blank=True, max_length=1000)), - ], - ), - ] diff --git a/collections_module/migrations/__init__.py b/common/templatetags/__init__.py similarity index 100% rename from collections_module/migrations/__init__.py rename to common/templatetags/__init__.py diff --git a/customtags/templatetags/can_create_app.py b/common/templatetags/can_create_app.py similarity index 62% rename from customtags/templatetags/can_create_app.py rename to common/templatetags/can_create_app.py index 56f490e5b..192099da0 100644 --- a/customtags/templatetags/can_create_app.py +++ b/common/templatetags/can_create_app.py @@ -1,6 +1,6 @@ from django import template -from apps.models import AppInstance +from apps.app_registry import APP_REGISTRY register = template.Library() @@ -10,6 +10,7 @@ def can_create_app(user, project, app): app_slug = app if isinstance(app, str) else app.slug - user_can_create = AppInstance.objects.user_can_create(user=user, project=project, app_slug=app_slug) + model_class = APP_REGISTRY.get_orm_model(app_slug) + user_can_create = model_class.objects.user_can_create(user=user, project=project, app_slug=app_slug) return user_can_create diff --git a/customtags/templatetags/get_dict_key.py b/common/templatetags/get_dict_key.py similarity index 100% rename from customtags/templatetags/get_dict_key.py rename to common/templatetags/get_dict_key.py diff --git a/customtags/templatetags/get_range.py b/common/templatetags/get_range.py similarity index 100% rename from customtags/templatetags/get_range.py rename to common/templatetags/get_range.py diff --git a/customtags/templatetags/get_setting.py b/common/templatetags/get_setting.py similarity index 100% rename from customtags/templatetags/get_setting.py rename to common/templatetags/get_setting.py diff --git a/customtags/templatetags/get_version.py b/common/templatetags/get_version.py similarity index 100% rename from customtags/templatetags/get_version.py rename to common/templatetags/get_version.py diff --git a/customtags/templatetags/is_login_signup_disabled.py b/common/templatetags/is_login_signup_disabled.py similarity index 100% rename from customtags/templatetags/is_login_signup_disabled.py rename to common/templatetags/is_login_signup_disabled.py diff --git a/customtags/__init__.py b/customtags/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/customtags/migrations/__init__.py b/customtags/migrations/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/customtags/templatetags/__init__.py b/customtags/templatetags/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/cypress/e2e/setup-scripts/seed_collections_user.py b/cypress/e2e/setup-scripts/seed_collections_user.py index 58f38ef36..bed4e8703 100755 --- a/cypress/e2e/setup-scripts/seed_collections_user.py +++ b/cypress/e2e/setup-scripts/seed_collections_user.py @@ -5,10 +5,12 @@ from django.conf import settings from django.contrib.auth.models import User +from django.db import transaction -from apps.helpers import create_app_instance +from apps.app_registry import APP_REGISTRY +from apps.helpers import create_instance_from_form from apps.models import Apps -from projects.models import Environment, Flavor, Project, ProjectTemplate +from projects.models import Flavor, Project, ProjectTemplate from projects.tasks import create_resources_from_template cypress_path = os.path.join(settings.BASE_DIR, "cypress/fixtures") @@ -17,36 +19,60 @@ with open(os.path.join(cypress_path, "users.json"), "r") as f: testdata = json.load(f) - userdata = testdata["collections_user"] - username = userdata["username"] - email = userdata["email"] - pwd = userdata["password"] +userdata = testdata["collections_user"] +username = userdata["username"] +email = userdata["email"] +pwd = userdata["password"] + +with transaction.atomic(): # Create a superuser because that's the one that can currently create collections superuser = User.objects.create_superuser(username, email, pwd) superuser.save() + project_template = ProjectTemplate.objects.get(pk=1) # Create a project for apps to be included in the collection project = Project.objects.create_project( - name="e2e-collections-test-proj", owner=superuser, description="e2e-collections-test-proj-desc" + name="e2e-collections-test-proj", + owner=superuser, + description="e2e-collections-test-proj-desc", + project_template=project_template, ) project.save() # Create an app to be included in a collection # create resources inside the project - project_template = ProjectTemplate.objects.get(pk=1) + create_resources_from_template(superuser.username, project.slug, project_template.template) # define variables needed app = Apps.objects.filter(slug="dashapp").order_by("-revision").first() flavor = Flavor.objects.filter(project=project).first() - environment = Environment.objects.filter(project=project).first() + + # define variables needed + app_slug = "dashapp" + data = { - "app_name": "collection-app-name", - "app_description": "collection-app-description", + "name": "collection-app-name", + "description": "collection-app-description", "flavor": str(flavor.pk), - "permission": "public", - "environment": str(environment.pk), + "access": "public", + "port": 8000, + "image": "some-image", + "source_code_url": "https://someurlthatdoesnotexist.com", } - # now create app - create_app_instance(superuser, project, app, app.settings, data=data) + + # Check if the model form tuple exists + if app_slug not in APP_REGISTRY: + raise ValueError(f"Form class not found for app slug {app_slug}") + + form_class = APP_REGISTRY.get_form_class(app_slug) + + # Create form + form = form_class(data, project_pk=project.pk) + + if form.is_valid(): + # now create app + create_instance_from_form(form, project, app_slug) + else: + raise ValueError(f"Form is invalid: {form.errors.as_data()}") diff --git a/cypress/e2e/setup-scripts/seed_contributor.py b/cypress/e2e/setup-scripts/seed_contributor.py index a99741694..0fc4506bc 100755 --- a/cypress/e2e/setup-scripts/seed_contributor.py +++ b/cypress/e2e/setup-scripts/seed_contributor.py @@ -6,7 +6,7 @@ from django.conf import settings from django.contrib.auth import get_user_model -from projects.models import Project +from projects.models import Project, ProjectTemplate User = get_user_model() @@ -26,10 +26,13 @@ user = User.objects.create_user(username, email, pwd) else: user = User.objects.get(username=email) + project_template = ProjectTemplate.objects.get(pk=1) # Check if project exists, otherwise, create it if not Project.objects.filter(name="e2e-delete-proj-test").exists(): - _ = Project.objects.create_project(name="e2e-delete-proj-test", owner=user, description="") + _ = Project.objects.create_project( + name="e2e-delete-proj-test", owner=user, description="", project_template=project_template + ) # Create the contributor's collaborator user co_userdata = testdata["contributor_collaborator"] @@ -44,4 +47,6 @@ # Check if project exists, otherwise, create it if not Project.objects.filter(name="e2e-collaborator-proj-test").exists(): - _ = Project.objects.create_project(name="e2e-collaborator-proj-test", owner=co_user, description="") + _ = Project.objects.create_project( + name="e2e-collaborator-proj-test", owner=co_user, description="", project_template=project_template + ) diff --git a/cypress/e2e/setup-scripts/seed_superuser.py b/cypress/e2e/setup-scripts/seed_superuser.py index 337622e1a..0e36e4399 100755 --- a/cypress/e2e/setup-scripts/seed_superuser.py +++ b/cypress/e2e/setup-scripts/seed_superuser.py @@ -5,54 +5,73 @@ from django.conf import settings from django.contrib.auth.models import User +from django.db import transaction -from apps.helpers import create_app_instance -from apps.models import Apps -from projects.models import Environment, Flavor, Project, ProjectTemplate +from apps.app_registry import APP_REGISTRY +from apps.helpers import create_instance_from_form +from projects.models import Flavor, Project, ProjectTemplate from projects.tasks import create_resources_from_template +from studio.utils import get_logger + +logger = get_logger(__name__) cypress_path = os.path.join(settings.BASE_DIR, "cypress/fixtures") -print(f"Now loading the json users file from fixtures path: {cypress_path}") # /app/cypress/fixtures +logger.info("RUNNING CYPRESS TESTS SETUP SCRIPTS") +logger.debug(f"Now loading the json users file from fixtures path: {cypress_path}") # /app/cypress/fixtures with open(os.path.join(cypress_path, "users.json"), "r") as f: testdata = json.load(f) - s_userdata = testdata["superuser"] - s_username = s_userdata["username"] - s_email = s_userdata["email"] - s_pwd = s_userdata["password"] - # Create the superuser - superuser = User.objects.create_superuser(s_username, s_email, s_pwd) - superuser.save() - - userdata = testdata["superuser_testuser"] - username = userdata["username"] - email = userdata["email"] - pwd = userdata["password"] - # Create the regular user +s_userdata = testdata["superuser"] +s_username = s_userdata["username"] +s_email = s_userdata["email"] +s_pwd = s_userdata["password"] +# Create the superuser +superuser = User.objects.create_superuser(s_username, s_email, s_pwd) +superuser.save() + +userdata = testdata["superuser_testuser"] +username = userdata["username"] +email = userdata["email"] +pwd = userdata["password"] + +with transaction.atomic(): + logger.debug("Creating regular user") user = User.objects.create_user(username, email, pwd) user.save() - # Create a dummy project belonging to the regular user to be inspected by the superuser tests + project_template = ProjectTemplate.objects.get(pk=1) + + logger.debug("Create a dummy project belonging to the regular user to be inspected by the superuser tests") project = Project.objects.create_project( - name="e2e-superuser-testuser-proj-test", owner=user, description="Description by regular user" + name="e2e-superuser-testuser-proj-test", + owner=user, + description="Description by regular user", + project_template=project_template, ) project.save() # Create a private app belonging to the regular user to be inspected by the superuser - # create resources inside the project - project_template = ProjectTemplate.objects.get(pk=1) + create_resources_from_template(user.username, project.slug, project_template.template) - # define variables needed - app = Apps.objects.filter(slug="jupyter-lab").order_by("-revision").first() + flavor = Flavor.objects.filter(project=project).first() - environment = Environment.objects.filter(project=project).first() - data = { - "app_name": "Regular user's private app", - "app_description": "Test app for superuser testing", - "flavor": str(flavor.pk), - "permission": "private", - "environment": str(environment.pk), - } - # now create app - create_app_instance(user, project, app, app.settings, data=data) + + # define variables needed + app_slug = "jupyter-lab" + + data = {"name": "Regular user's private app", "flavor": str(flavor.pk), "access": "private", "volume": None} + + if app_slug not in APP_REGISTRY: + raise ValueError(f"Form class not found for app slug {app_slug}") + + form_class = APP_REGISTRY.get_form_class(app_slug) + + # Create form + form = form_class(data, project_pk=project.pk) + + if form.is_valid(): + # now create app + create_instance_from_form(form, project, app_slug) + else: + raise ValueError(f"Form is invalid: {form.errors.as_data()}") diff --git a/cypress/e2e/ui-tests/test-collections.cy.js b/cypress/e2e/ui-tests/test-collections.cy.js index 381e01aad..46ba9eb07 100644 --- a/cypress/e2e/ui-tests/test-collections.cy.js +++ b/cypress/e2e/ui-tests/test-collections.cy.js @@ -50,7 +50,7 @@ describe("Test collections functionality", () => { cy.get('li.success').should('contain', "was added successfully") // confirm collection was created cy.log("Adding an app to the collection") - cy.get('tr.model-appinstance').find('a').first().click() + cy.get('tr.model-dashinstance').find('a').first().click() cy.get('tr').find('a').contains(collection_app_name).click() cy.get('#id_collections').select(collection_name) cy.get('input[name=_save]').click() diff --git a/cypress/e2e/ui-tests/test-deploy-app.cy.js b/cypress/e2e/ui-tests/test-deploy-app.cy.js index 85be502d5..6ae9e84d3 100644 --- a/cypress/e2e/ui-tests/test-deploy-app.cy.js +++ b/cypress/e2e/ui-tests/test-deploy-app.cy.js @@ -52,8 +52,8 @@ describe("Test deploying app", () => { const image_name_2 = "ghcr.io/scilifelabdatacentre/example-streamlit:230921-1443" const image_port = "8501" const image_port_2 = "8502" - const app_path = "username" - const app_path_2 = "username/app" + const app_path = "/home/username" + const app_path_2 = "/home/username/app" const link_privacy_type_note = "some-text-on-link-only-app" const createResources = Cypress.env('create_resources'); const app_type = "Custom App" @@ -66,13 +66,14 @@ describe("Test deploying app", () => { // Create an app with project permissions cy.log("Now creating a project app") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name_project) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('#permission').select('project') - cy.get('input[name="appconfig.port"]').clear().type(image_port) - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.path"]').clear().type(app_path) - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name_project) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Project') + cy.get('#id_volume').select('project-vol') + cy.get('#id_port').clear().type(image_port) + cy.get('#id_image').clear().type(image_name) + cy.get('#id_path').clear().type(app_path) + cy.get('#submit-id-submit').contains('Submit').click() // check that the app was created cy.get('tr:contains("' + app_name_project + '")').find('span').should('contain', 'Running') cy.get('tr:contains("' + app_name_project + '")').find('span').should('contain', 'project') @@ -87,9 +88,9 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name_project + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name_project + '")').find('a').contains('Settings').click() - cy.get('#permission').select('public') - cy.get('input[name=source_code_url]').type(app_source_code_public) - cy.get('button').contains('Update').click() + cy.get('#id_access').select('Public') + cy.get('#id_source_code_url').type(app_source_code_public) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name_project + '")').find('span').should('contain', 'Running') cy.get('tr:contains("' + app_name_project + '")').find('span').should('contain', 'public') @@ -102,15 +103,15 @@ describe("Test deploying app", () => { // Create a public app and verify that it is displayed on the public apps page cy.log("Now creating a public app") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name_public) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('#permission').select('public') - cy.get('input[name=source_code_url]').type(app_source_code_public) - cy.get('input[name="appconfig.port"]').clear().type(image_port) - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.path"]').clear().type(app_path) - cy.get('#Persistent\\ Volume').select('project-vol') - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name_public) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Public') + cy.get('#id_source_code_url').type(app_source_code_public) + cy.get('#id_port').clear().type(image_port) + cy.get('#id_image').clear().type(image_name) + cy.get('#id_path').clear().type(app_path) + cy.get('#id_volume').select('project-vol') + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name_public + '")').find('span').should('contain', 'Running') cy.get('tr:contains("' + app_name_public + '")').find('span').should('contain', 'public') @@ -151,24 +152,23 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name_public + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name_public + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').should('have.value', app_name_public) // name should be same as before - cy.get('input[name=app_name]').clear().type(app_name_public_2) // now change name - cy.get('textarea[name=app_description]').should('have.value', app_description) // description should be same as set before - cy.get('textarea[name=app_description]').clear().type(app_description_2) // now change description - cy.get('#permission').find(':selected').should('contain', 'public') - // checking that a) permissions can be changed to 'link'; b) that the corresponding text field is shown and mandatory - cy.get('#permission').select('link') - cy.get('textarea[name=link_privacy_type_note]').should('be.visible') - cy.get('textarea[name=link_privacy_type_note]').should('have.attr', 'required') - cy.get('textarea[name=link_privacy_type_note]').clear().type(link_privacy_type_note) - cy.get('#Persistent\\ Volume').find(':selected').should('contain', 'project-vol') - cy.get('input[name="appconfig.port"]').should('have.value', image_port) - cy.get('input[name="appconfig.port"]').clear().type(image_port_2) - cy.get('input[name="appconfig.image"]').should('have.value', image_name) - cy.get('input[name="appconfig.image"]').clear().type(image_name_2) - cy.get('input[name="appconfig.path"]').should('have.value', app_path) - cy.get('input[name="appconfig.path"]').clear().type(app_path_2) - cy.get('button').contains('Update').click() + cy.get('#id_name').should('have.value', app_name_public) // name should be same as before + cy.get('#id_name').clear().type(app_name_public_2) // now change name + cy.get('#id_description').should('have.value', app_description) // description should be same as set before + cy.get('#id_description').clear().type(app_description_2) // now change description + cy.get('#id_access').find(':selected').should('contain', 'Public') + // checking that a) permissions can be changed to 'Link'; b) that the corresponding text field is shown and mandatory + cy.get('#id_access').select('Link') + cy.get('#id_note_on_linkonly_privacy').should('be.visible') + cy.get('#id_note_on_linkonly_privacy').clear().type(link_privacy_type_note) + cy.get('#id_volume').find(':selected').should('contain', 'project-vol') + cy.get('#id_port').should('have.value', image_port) + cy.get('#id_port').clear().type(image_port_2) + cy.get('#id_image').should('have.value', image_name) + cy.get('#id_image').clear().type(image_name_2) + cy.get('#id_path').should('have.value', app_path) + cy.get('#id_path').clear().type(app_path_2) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name_public_2 + '")').find('span').should('contain', 'link') cy.get('tr:contains("' + app_name_public_2 + '")').find('span').should('contain', 'Running') // NB: it will get status "Running" but it won't work because the new port is incorrect // Check that the changes were saved @@ -176,13 +176,13 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name_public_2 + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name_public_2 + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').should('have.value', app_name_public_2) - cy.get('textarea[name=app_description]').should('have.value', app_description_2) - cy.get('#permission').find(':selected').should('contain', 'link') - cy.get('textarea[name=link_privacy_type_note]').should('have.value', link_privacy_type_note) - cy.get('input[name="appconfig.port"]').should('have.value', image_port_2) - cy.get('input[name="appconfig.image"]').should('have.value', image_name_2) - cy.get('input[name="appconfig.path"]').should('have.value', app_path_2) + cy.get('#id_name').should('have.value', app_name_public_2) + cy.get('#id_description').should('have.value', app_description_2) + cy.get('#id_access').find(':selected').should('contain', 'Link') + cy.get('#id_note_on_linkonly_privacy').should('have.value', link_privacy_type_note) + cy.get('#id_port').should('have.value', image_port_2) + cy.get('#id_image').should('have.value', image_name_2) + cy.get('#id_path').should('have.value', app_path_2) // Remove the created public app and verify that it is deleted from public apps page cy.log("Now deleting the public app") @@ -218,13 +218,13 @@ describe("Test deploying app", () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('input[name=source_code_url]').type(source_code_url) - cy.get('#permission').select('public') - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.port"]').clear().type(image_port) - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Public') + cy.get('#id_source_code_url').type(source_code_url) + cy.get('#id_image').clear().type(image_name) + cy.get('#id_port').clear().type(image_port) + cy.get('#submit-id-submit').contains('Submit').click() // cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'Running') // for now commented out because it takes shinyproxy a really long time to start up and therefore status "Running" can take 5 minutes to show up cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'public') @@ -233,11 +233,11 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').should('have.value', app_name) - cy.get('textarea[name=app_description]').should('have.value', app_description) - cy.get('#permission').find(':selected').should('contain', 'public') - cy.get('input[name="appconfig.image"]').should('have.value', image_name) - cy.get('input[name="appconfig.port"]').should('have.value', image_port) + cy.get('#id_name').should('have.value', app_name) + cy.get('#id_description').should('have.value', app_description) + cy.get('#id_access').find(':selected').should('contain', 'Public') + cy.get('#id_image').should('have.value', image_name) + cy.get('#id_port').should('have.value', image_port) cy.log("Checking that the shiny app is displayed on the public apps page") cy.visit("/apps") @@ -287,13 +287,13 @@ describe("Test deploying app", () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('input[name=source_code_url]').type(source_code_url) - cy.get('#permission').select('public') - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.port"]').clear().type(image_port) - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Public') + cy.get('#id_source_code_url').type(source_code_url) + cy.get('#id_image').clear().type(image_name) + cy.get('#id_port').clear().type(image_port) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'Running') cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'public') @@ -302,11 +302,11 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').should('have.value', app_name) - cy.get('textarea[name=app_description]').should('have.value', app_description) - cy.get('#permission').find(':selected').should('contain', 'public') - cy.get('input[name="appconfig.image"]').should('have.value', image_name) - cy.get('input[name="appconfig.port"]').should('have.value', image_port) + cy.get('#id_name').should('have.value', app_name) + cy.get('#id_description').should('have.value', app_description) + cy.get('#id_access').find(':selected').should('contain', 'Public') + cy.get('#id_image').should('have.value', image_name) + cy.get('#id_port').should('have.value', image_port) cy.log("Deleting the dash app") cy.visit("/projects/") @@ -338,11 +338,11 @@ describe("Test deploying app", () => { cy.log("Creating a tisuumaps app") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('#permission').select('public') - cy.get('#Persistent\\ Volume').select('project-vol') - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Public') + cy.get('#id_volume').select('project-vol') + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'Running') cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'public') @@ -351,10 +351,10 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').should('have.value', app_name) - cy.get('textarea[name=app_description]').should('have.value', app_description) - cy.get('#permission').find(':selected').should('contain', 'public') - cy.get('#Persistent\\ Volume').find(':selected').should('contain', 'project-vol') + cy.get('#id_name').should('have.value', app_name) + cy.get('#id_description').should('have.value', app_description) + cy.get('#id_access').find(':selected').should('contain', 'Public') + cy.get('#id_volume').find(':selected').should('contain', 'project-vol') cy.log("Deleting the tissuumaps app") cy.visit("/projects/") @@ -376,6 +376,7 @@ describe("Test deploying app", () => { // Names of objects to create const project_name = "e2e-deploy-app-test" const app_name = "e2e-subdomain-example" + const app_name_2 = "e2e-second-subdomain-example" const app_description = "e2e-subdomain-description" const image_name = "ghcr.io/scilifelabdatacentre/example-streamlit:latest" const createResources = Cypress.env('create_resources'); @@ -391,36 +392,41 @@ describe("Test deploying app", () => { cy.log("Now creating an app with a custom subdomain") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() // fill out other fields - cy.get('input[name=app_name]').type(app_name) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('input[name="appconfig.port"]').clear().type("8501") - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.path"]').clear().type("/home") + cy.get('#id_name').type(app_name) + cy.get('#id_description').type(app_description) + cy.get('#id_port').clear().type("8501") + cy.get('#id_image').clear().type(image_name) + cy.get('#id_volume').select('project-vol') + cy.get('#id_path').clear().type("/home") // fill out subdomain field - cy.get('[id="subdomain"]').find('button').click() - cy.get('[id="subdomain-add"]').find('[id="rn"]').type(subdomain) - cy.get('[id="subdomain-add"]').find('button').click() - cy.wait(5000) - cy.get('[id="subdomain"]').find('select#app_release_name option:selected').should('contain', subdomain) + cy.get('#id_subdomain').type(subdomain) + // create the app - cy.get('button').contains('Create').click() + cy.get('#submit-id-submit').contains('Submit').click() // check that the app was created with the correct subdomain cy.get('a').contains(app_name).should('have.attr', 'href').and('include', subdomain) // Try using the same subdomain the second time cy.log("Now trying to create an app with an already taken subdomain") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() + + cy.get('#id_name').type(app_name_2) + cy.get('#id_port').clear().type("8501") + cy.get('#id_image').clear().type(image_name) + // fill out subdomain field - cy.get('[id="subdomain"]').find('button').click() - cy.get('[id="subdomain-add"]').find('[id="rn"]').type(subdomain) - cy.get('[id="subdomain-add"]').find('button').click() - cy.wait(5000) - cy.get('[id="subdomain-add"]').find('[id="subdomain-invalid"]').should('be.visible') // display errror when same subdomain - cy.get('[id="subdomain-add"]').find('[id="rn"]').clear().type(subdomain_2) - cy.get('[id="subdomain-add"]').find('button').click() - cy.wait(5000) - cy.get('[id="subdomain-add"]').find('[id="subdomain-invalid"]').should('not.be.visible') // do not display error when different subdomain - cy.get('[id="subdomain"]').find('select#app_release_name option:selected').should('contain', subdomain_2) // and the newly added subdomain as selected again + cy.get('#id_subdomain').type(subdomain) + cy.get('#id_subdomain').blur(); + cy.get('#div_id_subdomain').should('contain.text', 'The subdomain is not available'); + + + cy.get('#id_subdomain').clear().type(subdomain_2) + cy.get('#id_subdomain').blur(); + cy.get('#div_id_subdomain').should('contain.text', 'The subdomain is available'); + // create the app + cy.get('#submit-id-submit').contains('Submit').click() + // check that the app was created with the correct subdomain + cy.get('a').contains(app_name_2).should('have.attr', 'href').and('include', subdomain_2) // Change subdomain of a previously created app cy.log("Now changing subdomain of an already created app") @@ -428,12 +434,9 @@ describe("Test deploying app", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains("Settings").click() - cy.get('[id="subdomain"]').find('button').click() - cy.get('[id="subdomain-add"]').find('[id="rn"]').type(subdomain_3) - cy.get('[id="subdomain-add"]').find('button').click() - cy.wait(5000) - // update the app - cy.get('button').contains('Update').click() + cy.get('#id_subdomain').clear().type(subdomain_3) + + cy.get('#submit-id-submit').contains('Submit').click() // check that the app was updated with the correct subdomain cy.get('a').contains(app_name).should('have.attr', 'href').and('include', subdomain_3) @@ -460,19 +463,19 @@ describe("Test deploying app", () => { // Create an app with project permissions cy.log("Now creating an app with a non-existent image reference - expecting Image Error") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name_statuses) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('#permission').select('project') - cy.get('input[name="appconfig.port"]').type("8501") - cy.get('input[name="appconfig.image"]').type("hkqxqxkhkqwxhkxwh") // input random string - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name_statuses) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Project') + cy.get('#id_port').type("8501") + cy.get('#id_image').type("hkqxqxkhkqwxhkxwh") // input random string + cy.get('#submit-id-submit').contains('Submit').click() // check that the app was created cy.get('tr:contains("' + app_name_statuses + '")').find('span').should('contain', 'Image Error') cy.log("Now updating the app to give a correct image reference - expecting Running") cy.get('tr:contains("' + app_name_statuses + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name_statuses + '")').find('a').contains('Settings').click() - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('button').contains('Update').click() + cy.get('#id_image').clear().type(image_name) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name_statuses + '")').find('span').should('contain', 'Running') } else { cy.log('Skipped because create_resources is not true'); diff --git a/cypress/e2e/ui-tests/test-project-as-contributor.cy.js b/cypress/e2e/ui-tests/test-project-as-contributor.cy.js index db8a9cbaa..695948fad 100644 --- a/cypress/e2e/ui-tests/test-project-as-contributor.cy.js +++ b/cypress/e2e/ui-tests/test-project-as-contributor.cy.js @@ -228,8 +228,8 @@ describe("Test project contributor user functionality", () => { // step 1. create 3 jupyter lab instances (current limit) Cypress._.times(3, () => { cy.get('[data-cy="create-app-card"]').contains('Jupyter Lab').parent().siblings().find('.btn').click() - cy.get('input[name=app_name]').type("e2e-create-jl") - cy.get('.btn-primary').contains('Create').click() + cy.get('#id_name').type("e2e-create-jl") + cy.get('#submit-id-submit').contains('Submit').click() }); // step 2. check that the button to create another one does not work cy.get('[data-cy="create-app-card"]').contains('Jupyter Lab').parent().siblings().find('.btn').should('not.have.attr', 'href') @@ -357,17 +357,17 @@ describe("Test project contributor user functionality", () => { // Create private app cy.log("Now creating a private app") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(private_app_name) - cy.get('select[id=permission]').select('private') - cy.get('button').contains('Create').click() // create app + cy.get('#id_name').type(private_app_name) + cy.get('#id_access').select('Private') + cy.get('#submit-id-submit').contains('Submit').click() // create app cy.get('tr:contains("' + private_app_name + '")').find('span').should('contain', 'private') // check that the app got greated // Create project app cy.log("Now creating a project app") cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(project_app_name) - cy.get('select[id=permission]').select('project') - cy.get('button').contains('Create').click() // create app + cy.get('#id_name').type(project_app_name) + cy.get('#id_access').select('Project') + cy.get('#submit-id-submit').contains('Submit').click() // create app cy.get('tr:contains("' + project_app_name + '")').find('span').should('contain', 'project') // check that the app got greated // Give access to this project to a collaborator user @@ -453,7 +453,7 @@ describe("Test project contributor user functionality", () => { }) }) - it("can create a file management instance", () => { + it("can create a file management instance", { defaultCommandTimeout: 100000 }, () => { const project_name = "e2e-create-proj-test" cy.log("Creating a blank project") @@ -463,13 +463,9 @@ describe("Test project contributor user functionality", () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() - cy.get('div.card-body:contains("Activate File Manager")').find('a:contains("Activate")').click() - cy.get('button').contains("Activate").first().click() - cy.get('#manage-files > .card > .row').should('contain', 'File Manager is activated') + cy.get('div.card-body:contains("File Manager")').find('a:contains("Create")').click() + cy.get('#submit-id-submit').click() - // change the command to check for Created, Pending or Running - cy.get('#manage-files .card-header').find('span.badge').invoke('text').then((text) => { - expect(["Created", "Pending", "Running"]).to.include(text.trim()); - }); + cy.get('tr:contains("File Manager")').find('span').should('contain', 'Running') }) }) diff --git a/cypress/e2e/ui-tests/test-superuser-functionality.cy.js b/cypress/e2e/ui-tests/test-superuser-functionality.cy.js index 96e839a21..e093ca364 100644 --- a/cypress/e2e/ui-tests/test-superuser-functionality.cy.js +++ b/cypress/e2e/ui-tests/test-superuser-functionality.cy.js @@ -60,11 +60,6 @@ describe("Test superuser access", () => { cy.get('h3').should('contain', project_name) cy.get('.card-text').should('contain', project_description) - cy.log("Checking that the correct deployment options are available (i.e. with extra deployment options)") - cy.get('.card-header').find('h5').should('contain', 'Develop') - cy.get('.card-header').find('h5').should('contain', 'Serve') - cy.get('.card-body').find('h5').should('contain', 'Shiny App (single copy)') - cy.get('.card-header').find('h5').should('contain', 'Additional options [admins only]') cy.log("Checking that project settings are available") cy.get('[data-cy="settings"]').click() @@ -73,8 +68,6 @@ describe("Test superuser access", () => { cy.log("Checking that the correct project settings are visible (i.e. with extra settings)") cy.get('.list-group').find('a').should('contain', 'Access') - cy.get('.list-group').find('a').should('contain', 'S3 storage') - cy.get('.list-group').find('a').should('contain', 'MLFlow') cy.get('.list-group').find('a').should('contain', 'Flavors') cy.get('.list-group').find('a').should('contain', 'Environments') @@ -132,8 +125,8 @@ describe("Test superuser access", () => { cy.log("Verifying that can edit the private app of a regular user") cy.get('tr:contains("' + private_app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + private_app_name + '")').find('a').contains('Settings').click() - cy.get('input[name=app_name]').clear().type(private_app_name_2) // change name - cy.get('button').contains('Update').click() + cy.get('#id_name').clear().type(private_app_name_2) // change name + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + private_app_name_2 + '")').should('exist') // regular user's private app now has a different name cy.log("Deleting a regular user's private app") @@ -208,13 +201,13 @@ describe("Test superuser access", () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('div.card-body:contains("' + app_type + '")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('textarea[name=app_description]').type(app_description) - cy.get('#permission').select('project') - cy.get('#flavor').select('2 vCPU, 4 GB RAM') - cy.get('input[name="appconfig.image"]').clear().type(image_name) - cy.get('input[name="appconfig.port"]').clear().type(image_port) - cy.get('button').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#id_description').type(app_description) + cy.get('#id_access').select('Project') + cy.get('#id_flavor').select('2 vCPU, 4 GB RAM') + cy.get('#id_image').clear().type(image_name) + cy.get('#id_port').clear().type(image_port) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'Running') cy.log("Changing the flavor setting") @@ -222,9 +215,9 @@ describe("Test superuser access", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains('Settings').click() - cy.get('#flavor').find(':selected').should('contain', '2 vCPU, 4 GB RAM') - cy.get('#flavor').select(new_flavor_name) - cy.get('button').contains('Update').click() + cy.get('#id_flavor').find(':selected').should('contain', '2 vCPU, 4 GB RAM') + cy.get('#id_flavor').select(new_flavor_name) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name + '")').find('span').should('contain', 'Running') cy.log("Checking that the new flavor setting was saved in the database") @@ -232,7 +225,7 @@ describe("Test superuser access", () => { cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('tr:contains("' + app_name + '")').find('i.bi-three-dots-vertical').click() cy.get('tr:contains("' + app_name + '")').find('a').contains('Settings').click() - cy.get('#flavor').find(':selected').should('contain', new_flavor_name) + cy.get('#id_flavor').find(':selected').should('contain', new_flavor_name) } else { cy.log('Skipped because create_resources is not true'); @@ -249,7 +242,10 @@ describe("Test superuser access", () => { }) }) - it("can create a persistent volume", () => { + it.skip("can create a persistent volume", () => { + // This test is not used, since creating PVCs like this is not the correct way any more. + // The correct way is to create a volume in the admin panel. + // Names of objects to create const project_name_pvc = "e2e-superuser-pvc-test" const volume_name = "e2e-project-vol" @@ -262,8 +258,8 @@ describe("Test superuser access", () => { cy.contains('.card-title', project_name_pvc).parents('.card-body').siblings('.card-footer').find('a:contains("Open")').first().click() cy.get('div.card-body:contains("Persistent Volume")').find('a:contains("Create")').click() - cy.get('input[name=app_name]').type(volume_name) - cy.get('button').contains('Create').click() + cy.get('#id_name').type(volume_name) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + volume_name + '")').should('exist') // persistent volume has been created // This does not work in our CI. Disabled for now, needs to be enabled for runs against an instance of Serve running on the cluster @@ -337,15 +333,15 @@ describe("Test superuser access", () => { cy.log("Create 3 jupyter lab instances (current limit)") Cypress._.times(3, () => { cy.get('[data-cy="create-app-card"]').contains('Jupyter Lab').parent().siblings().find('.btn').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('.btn-primary').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#submit-id-submit').contains('Submit').click() }); cy.log("Check that the button to create another one still works") cy.get('[data-cy="create-app-card"]').contains('Jupyter Lab').parent().siblings().find('.btn').should('have.attr', 'href') cy.log("Check that it is possible to create another one and therefore bypass the limit") cy.get('[data-cy="create-app-card"]').contains('Jupyter Lab').parent().siblings().find('.btn').click() - cy.get('input[name=app_name]').type(app_name) - cy.get('.btn-primary').contains('Create').click() + cy.get('#id_name').type(app_name) + cy.get('#submit-id-submit').contains('Submit').click() cy.get('tr:contains("' + app_name + '")').its('length').should('eq', 4) // we now have an extra app }) diff --git a/fixtures/apps_fixtures.json b/fixtures/apps_fixtures.json index 2a10a45ae..7d4469e90 100644 --- a/fixtures/apps_fixtures.json +++ b/fixtures/apps_fixtures.json @@ -2,47 +2,12 @@ { "fields": { "category": "develop", - "chart": "ghcr.io/scilifelabdatacentre/serve-charts/lab:1.0.0", + "chart": "ghcr.io/scilifelabdatacentre/serve-charts/lab:1.0.1", "created_on": "2021-02-19T21:34:37.815Z", "description": "", "logo": "jupyter-lab-logo.svg", "name": "Jupyter Lab", "priority": "500", - "settings": { - "apps": { - "Persistent Volume": "many" - }, - "default_values": { - "port": "80", - "targetport": "8888" - }, - "environment": { - "name": "from", - "quantity": "one", - "title": "Image", - "type": "match" - }, - "export-cli": "True", - "flavor": "one", - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "false", - "value": "false" - } - } - }, "slug": "jupyter-lab", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -61,33 +26,6 @@ "logo": "filemanager-logo.svg", "name": "File Manager", "priority": "200", - "settings": { - "apps": { - "Persistent Volume": "one" - }, - "default_values": { - "port": "8080", - "targetport": "8080" - }, - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "false", - "value": "false" - } - } - }, "slug": "filemanager", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -325,47 +263,6 @@ "description": "", "name": "Persistent Volume", "priority": "600", - "settings": { - "default_values": { - "port": "port", - "targetport": "targetport" - }, - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "false", - "value": "false" - } - }, - "volume": { - "accessModes": { - "default": "ReadWriteMany", - "title": "AccessModes", - "type": "string" - }, - "size": { - "default": "1Gi", - "title": "Size", - "type": "string" - }, - "storageClass": { - "default": "", - "title": "StorageClass", - "type": "string" - } - } - }, "slug": "volumeK8s", "table_field": {}, "updated_on": "2021-03-10T19:45:03.927Z" @@ -381,35 +278,6 @@ "description": "", "logo": "vscode-logo.svg", "name": "VS Code", - "settings": { - "apps": { - "Persistent Volume": "many" - }, - "default_values": { - "port": "80", - "targetport": "8080" - }, - "export-cli": "True", - "flavor": "one", - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "false", - "value": "false" - } - } - }, "slug": "vscode", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -449,63 +317,6 @@ "logo": "default-logo.svg", "name": "Custom App", "priority": "400", - "settings": { - "appconfig": { - "image": { - "default": "registry/repository/image:tag", - "title": "Image", - "type": "string" - }, - "meta": { - "title": "Docker image configurations" - }, - "path": { - "default": "", - "title": "Path to folder for persistent storage (leave blank if Persistent Volume is set to None)", - "type": "string" - }, - "port": { - "default": "8000", - "title": "Port", - "type": "number" - }, - "userid": { - "default": "1000", - "title": "User ID", - "type": "number" - } - }, - "apps": { - "Persistent Volume": "one" - }, - "default_values": { - "port": "80", - "targetport": "8000" - }, - "flavor": "one", - "logs": [ - "serve" - ], - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "true", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - }, - "publishable": "true" - }, "slug": "customapp", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -524,34 +335,6 @@ "logo": "rstudio-logo.svg", "name": "RStudio", "priority": "600", - "settings": { - "apps": { - "Persistent Volume": "many" - }, - "default_values": { - "port": "80", - "targetport": "8787" - }, - "flavor": "one", - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "false", - "value": "false" - } - } - }, "slug": "rstudio", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -570,50 +353,6 @@ "logo": "dashapp-logo.svg", "name": "Dash App", "priority": "400", - "settings": { - "appconfig": { - "image": { - "default": "registry/repository/image:tag", - "title": "Image", - "type": "string" - }, - "meta": { - "title": "Docker image onfigurations" - }, - "port": { - "default": "8000", - "title": "Port", - "type": "number" - } - }, - "default_values": { - "port": "80", - "targetport": "8000" - }, - "flavor": "one", - "logs": [ - "serve" - ], - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "true", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - }, - "publishable": "true" - }, "slug": "dashapp", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -626,56 +365,12 @@ { "fields": { "category": "serve", - "chart": "ghcr.io/scilifelabdatacentre/serve-charts/shinyapp:1.0.1", + "chart": "ghcr.io/scilifelabdatacentre/serve-charts/shinyapp:1.0.2", "created_on": "2023-08-25T21:34:37.815Z", - "description": "Shiny app without ShinyProxy as a middle layer. [admins only]", + "description": "", "logo": "shinyapp-logo.svg", - "name": "Shiny App (single copy)", + "name": "Shiny App", "priority": "400", - "settings": { - "appconfig": { - "image": { - "default": "registry/repository/image:tag", - "title": "Image", - "type": "string" - }, - "meta": { - "title": "Docker image configurations" - }, - "port": { - "default": "3838", - "title": "Port", - "type": "number" - } - }, - "default_values": { - "port": "80", - "targetport": "3838" - }, - "flavor": "one", - "logs": [ - "serve" - ], - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "true", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - }, - "publishable": "true" - }, "slug": "shinyapp", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -688,71 +383,12 @@ { "fields": { "category": "serve", - "chart": "ghcr.io/scilifelabdatacentre/serve-charts/shinyproxy:1.1.0", + "chart": "ghcr.io/scilifelabdatacentre/serve-charts/shinyproxy:1.0.0", "created_on": "2023-08-25T21:34:37.815Z", "description": "", "logo": "shinyapp-logo.svg", - "name": "Shiny App", + "name": "ShinyProxy App", "priority": "400", - "settings": { - "appconfig": { - "image": { - "default": "registry/repository/image:tag", - "title": "Image", - "type": "string" - }, - "meta": { - "title": "Docker image configurations" - }, - "port": { - "default": "3838", - "title": "Port", - "type": "number" - }, - "proxyheartbeatrate": { - "default": "10000", - "title": "Proxy heartbeat rate", - "type": "number" - }, - "proxyheartbeattimeout": { - "default": "60000", - "title": "Proxy heartbeat timeout", - "type": "number" - }, - "proxycontainerwaittime": { - "default": "30000", - "title": "Proxy container wait time", - "type": "number" - } - }, - "default_values": { - "port": "80", - "targetport": "3838" - }, - "flavor": "one", - "logs": [ - "serve" - ], - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "true", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - }, - "publishable": "true" - }, "slug": "shinyproxyapp", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" @@ -831,52 +467,6 @@ "model": "apps.apps", "pk": 26 }, - { - "fields": { - "category": "admin-apps", - "chart": "ghcr.io/scilifelabdatacentre/serve-charts/filemanager:1.0.2", - "created_on": "2023-02-19T21:34:37.815Z", - "description": "", - "logo": "filemanager-logo.svg", - "name": "File Manager Admin", - "priority": "200", - "settings": { - "apps": { - "Persistent Volume": "one" - }, - "default_values": { - "port": "8080", - "targetport": "8080" - }, - "flavor": "one", - "permissions": { - "private": { - "option": "true", - "value": "false" - }, - "project": { - "option": "true", - "value": "true" - }, - "public": { - "option": "false", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - } - }, - "slug": "filemanager-admin", - "table_field": { - "url": "https://{{ release }}.{{ global.domain }}" - }, - "updated_on": "2023-03-10T19:45:03.927Z" - }, - "model": "apps.apps", - "pk": 27 - }, { "fields": { "category": "serve", @@ -886,38 +476,6 @@ "logo": "tissuumaps-logo.svg", "name": "TissUUmaps App", "priority": "400", - "settings": { - "apps": { - "Persistent Volume": "one" - }, - "default_values": { - "port": "80", - "targetport": "80" - }, - "flavor": "one", - "logs": [ - "serve" - ], - "permissions": { - "private": { - "option": "true", - "value": "true" - }, - "project": { - "option": "true", - "value": "false" - }, - "public": { - "option": "true", - "value": "false" - }, - "link": { - "option": "true", - "value": "false" - } - }, - "publishable": "true" - }, "slug": "tissuumaps", "table_field": { "url": "https://{{ release }}.{{ global.domain }}" diff --git a/fixtures/periodic_tasks_fixtures.json b/fixtures/periodic_tasks_fixtures.json index 5c3f3ef53..939409ab9 100644 --- a/fixtures/periodic_tasks_fixtures.json +++ b/fixtures/periodic_tasks_fixtures.json @@ -27,62 +27,7 @@ "model": "django_celery_beat.periodictask", "pk": 1 }, - { - "fields": { - "args": "[]", - "clocked": null, - "crontab": null, - "date_changed": "2021-02-26T14:03:40.168Z", - "description": "", - "enabled": true, - "exchange": null, - "expire_seconds": null, - "expires": null, - "headers": "{}", - "interval": 1, - "kwargs": "{}", - "last_run_at": "2021-02-26T14:03:37.169Z", - "name": "sync_mlflow_models", - "one_off": false, - "priority": null, - "queue": null, - "routing_key": null, - "solar": null, - "start_time": null, - "task": "apps.tasks.sync_mlflow_models", - "total_run_count": 174 - }, - "model": "django_celery_beat.periodictask", - "pk": 4 - }, - { - "fields": { - "args": "[]", - "clocked": null, - "crontab": 1, - "date_changed": "2021-02-26T14:03:40.168Z", - "description": "", - "enabled": true, - "exchange": null, - "expire_seconds": null, - "expires": null, - "headers": "{}", - "interval": null, - "kwargs": "{}", - "last_run_at": "2021-02-26T14:03:37.169Z", - "name": "purge_tasks", - "one_off": false, - "priority": null, - "queue": null, - "routing_key": null, - "solar": null, - "start_time": null, - "task": "apps.tasks.purge_tasks", - "total_run_count": 174 - }, - "model": "django_celery_beat.periodictask", - "pk": 5 - }, + { "fields": { "args": "[]", diff --git a/fixtures/projects_templates.json b/fixtures/projects_templates.json index 44c959664..6a3a27237 100644 --- a/fixtures/projects_templates.json +++ b/fixtures/projects_templates.json @@ -4,21 +4,12 @@ "description": "Use this template if you intend to only deploy apps or use integrated development environments (IDEs).", "name": "Default project", "slug": "blank", + "available_apps": [8, 9, 19, 21, 22, 23, 24, 28], "template": { "apps": { - "project-netpolicy": { - "permission": "private", - "slug": "netpolicy" - }, - "project-vol": { - "app_id": "", - "app_release_name": "", - "permission": "project", - "slug": "volumeK8s", - "storageClass": "microk8s-hostpath", - "volume.accessModes": "ReadWriteMany", - "volume.size": "1Gi", - "volume.storageClass": "microk8s-hostpath" + + "netpolicy": { + "name": "project-netpolicy" } }, "environments": { @@ -38,6 +29,11 @@ "repository": "ghcr.io/scilifelabdatacentre" } }, + "volumes": { + "project-vol": { + "size": "1" + } + }, "flavors": { "2 vCPU, 4 GB RAM": { "cpu": { @@ -70,26 +66,22 @@ "slug": "default", "template": { "apps": { - "project-netpolicy": { - "permission": "private", - "slug": "netpolicy" + "netpolicy": { + "name": "project-netpolicy" }, - "project-vol": { - "app_id": "", - "app_release_name": "", - "permission": "project", - "slug": "volumeK8s", - "storageClass": "microk8s-hostpath", - "volume.accessModes": "ReadWriteMany", - "volume.size": "1Gi", - "volume.storageClass": "microk8s-hostpath" + "jupyter-lab": { + "name": "My Jupyter Lab", + "description": "Deployed via project template", + "volume": "project-vol", + "access": "private", + "flavor": "2 vCPU, 4 GB RAM" }, - "project-filemanager": { - "app:volumeK8s": [ - "project-vol" - ], - "permission": "project", - "slug": "filemanager-admin" + "filemanager": { + "name": "project-filemanager", + "volume": "project-vol", + "persistent": "True", + "flavor": "2 vCPU, 4 GB RAM", + "access": "private" } }, "environments": { @@ -109,6 +101,11 @@ "repository": "ghcr.io/scilifelabdatacentre" } }, + "volumes": { + "project-vol": { + "size": "1" + } + }, "flavors": { "2 vCPU, 4 GB RAM": { "cpu": { diff --git a/models/.github/dependabot.yml b/models/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/models/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/models/.github/workflows/branch-name-check.yaml b/models/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 03c9b39bf..000000000 --- a/models/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v2 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/models/.github/workflows/code-checks.yaml b/models/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/models/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/models/.gitignore b/models/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/models/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/models/LICENSE b/models/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/models/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/models/MANIFEST.in b/models/MANIFEST.in deleted file mode 100644 index f621c09e4..000000000 --- a/models/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include models/templates * diff --git a/models/README.md b/models/README.md deleted file mode 100644 index 7efda4a3d..000000000 --- a/models/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-models - -Models Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "models" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/studio/migrations/models/0001_initial.py b/models/migrations/0001_initial.py similarity index 96% rename from studio/migrations/models/0001_initial.py rename to models/migrations/0001_initial.py index eafcdc77d..bde819f25 100644 --- a/studio/migrations/models/0001_initial.py +++ b/models/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 +# Generated by Django 5.0.2 on 2024-05-27 07:28 import django.db.models.deletion import django.db.models.manager @@ -26,26 +26,19 @@ class Migration(migrations.Migration): ], ), migrations.CreateModel( - name="Tagulous_Model_tags", + name="Metadata", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("name", models.CharField(max_length=255, unique=True)), - ("slug", models.SlugField()), - ( - "count", - models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), - ), - ( - "protected", - models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), - ), + ("run_id", models.CharField(max_length=32)), + ("trained_model", models.CharField(default="", max_length=32)), + ("project", models.CharField(default="", max_length=255)), + ("model_details", models.TextField(blank=True)), + ("parameters", models.TextField(blank=True)), + ("metrics", models.TextField(blank=True)), ], options={ - "ordering": ("name",), - "abstract": False, - "unique_together": {("slug",)}, + "unique_together": {("run_id", "trained_model")}, }, - bases=(tagulous.models.models.BaseTagModel, models.Model), ), migrations.CreateModel( name="ModelLog", @@ -73,19 +66,26 @@ class Migration(migrations.Migration): }, ), migrations.CreateModel( - name="Metadata", + name="Tagulous_Model_tags", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("run_id", models.CharField(max_length=32)), - ("trained_model", models.CharField(default="", max_length=32)), - ("project", models.CharField(default="", max_length=255)), - ("model_details", models.TextField(blank=True)), - ("parameters", models.TextField(blank=True)), - ("metrics", models.TextField(blank=True)), + ("name", models.CharField(max_length=255, unique=True)), + ("slug", models.SlugField()), + ( + "count", + models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), + ), + ( + "protected", + models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), + ), ], options={ - "unique_together": {("run_id", "trained_model")}, + "ordering": ("name",), + "abstract": False, + "unique_together": {("slug",)}, }, + bases=(tagulous.models.models.BaseTagModel, models.Model), ), migrations.CreateModel( name="Model", @@ -131,7 +131,6 @@ class Migration(migrations.Migration): to="projects.environment", ), ), - ("object_type", models.ManyToManyField(blank=True, to="models.objecttype")), ( "project", models.ForeignKey( @@ -141,12 +140,7 @@ class Migration(migrations.Migration): to="projects.project", ), ), - ( - "s3", - models.ForeignKey( - blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.s3" - ), - ), + ("object_type", models.ManyToManyField(blank=True, to="models.objecttype")), ( "tags", tagulous.models.fields.TagField( diff --git a/models/models.py b/models/models.py index 7c7662f26..2a861d76a 100644 --- a/models/models.py +++ b/models/models.py @@ -99,7 +99,7 @@ class Model(models.Model): resource = models.URLField(max_length=2048, null=True, blank=True) object_type = models.ManyToManyField(ObjectType, blank=True) url = models.URLField(max_length=512, null=True, blank=True) - s3 = models.ForeignKey("projects.S3", null=True, blank=True, on_delete=models.CASCADE) + bucket = models.CharField(max_length=200, null=True, blank=True, default="models") path = models.CharField(max_length=200, null=True, blank=True, default="models") uploaded_at = models.DateTimeField(auto_now_add=True) diff --git a/models/setup.py b/models/setup.py deleted file mode 100644 index d45fa4f15..000000000 --- a/models/setup.py +++ /dev/null @@ -1,33 +0,0 @@ -from setuptools import setup - -setup( - name="studio-models", - version="0.0.1", - description="""Django app for handling portal in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["models"], - package_dir={"models": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.1", - "requests==2.31.0", - "django-guardian==2.4.0", - "Pillow==9.4.0", - "Markdown==3.4.1", - "django-tagulous==1.3.3", - "minio==7.0.2", - "s3fs==2022.1.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/models/tests.py b/models/tests.py index 1ec96e7d7..e69de29bb 100644 --- a/models/tests.py +++ b/models/tests.py @@ -1,418 +0,0 @@ -import os - -import boto3 -import pytest -from django.contrib.auth import get_user_model -from django.test import RequestFactory, TestCase -from django.urls import reverse -from moto.server import ThreadedMotoServer - -from projects.models import S3, Project - -from . import views -from .models import Model, ObjectType - -User = get_user_model() - -os.environ["AWS_ACCESS_KEY_ID"] = "testing" -os.environ["AWS_SECRET_ACCESS_KEY"] = "testing" -os.environ["AWS_SECURITY_TOKEN"] = "testing" -os.environ["AWS_SESSION_TOKEN"] = "testing" -os.environ["AWS_DEFAULT_REGION"] = "us-east-1" - -test_user = {"username": "foo1", "email": "foo@test.com", "password": "bar"} - - -class ModelViewTests(TestCase): - bucket_name = "test-bucket" - - def setUp(self): - # Set up mocked S3 - self.server = ThreadedMotoServer(port=5000) - self.server.start() - - s3 = boto3.resource("s3", endpoint_url="http://localhost:5000") - bucket = s3.Bucket(self.bucket_name) - bucket.create() - s3.meta.client.put_object(Bucket=self.bucket_name, Key="public_uid", Body=b"test") - s3.meta.client.put_object(Bucket=self.bucket_name, Key="test_uid", Body=b"test") - # Create user - self.user = User.objects.create_user(test_user["username"], test_user["email"], test_user["password"]) - - self.project = Project.objects.create_project(name="test-perm", owner=self.user, description="") - - new_model = Model( - uid="test_uid", - name="test", - bucket=self.bucket_name, - description="model_description", - model_card="", - project=self.project, - access="PR", - s3=S3.objects.create( - access_key=os.environ.get("AWS_ACCESS_KEY_ID"), - host="localhost:5000", - secret_key=os.environ.get("AWS_SECRET_ACCESS_KEY"), - owner=self.user, - project=self.project, - ), - ) - new_model.save() - self.private_model = new_model - public_model = Model( - uid="public_uid", - name="public", - bucket=self.bucket_name, - description="model_description", - model_card="", - project=self.project, - access="PR", - s3=S3.objects.create( - access_key=os.environ.get("AWS_ACCESS_KEY_ID"), - host="localhost:5000", - secret_key=os.environ.get("AWS_SECRET_ACCESS_KEY"), - owner=self.user, - project=self.project, - ), - ) - public_model.save() - self.client.login(username=test_user["email"], password="bar") - self.client.post( - reverse( - "models:publish_model", - kwargs={ - "user": self.user, - "project": self.project.slug, - "id": public_model.id, - }, - ) - ) - self.client.post( - reverse( - "models:publish_model", - kwargs={ - "user": self.user, - "project": self.project.slug, - "id": new_model.id, - }, - ) - ) - self.client.post( - reverse( - "models:unpublish_model", - kwargs={ - "user": self.user, - "project": self.project.slug, - "id": new_model.id, - }, - ) - ) - - def tearDown(self): - self.server.stop() - - def test_models_view(self): - # Get correct request - request = RequestFactory().get(reverse("models:index")) - response = views.index(request, user=self.user, project=self.project) - - # Check status code - assert response.status_code == 200 - assert "Models | SciLifeLab Serve (beta)" in response.content.decode() - - @pytest.mark.skip(reason="It's not working") - def test_public_model_details_view(self): - response = self.client.get(reverse("models:details_public", kwargs={"id": 1})) - - # Check status code - assert response.status_code == 200 - self.assertTemplateUsed(response, "models/models_details_public.html") - assert "Model Details" in response.content.decode() - assert "Model public Details | SciLifeLab Serve (beta)" in response.content.decode() - - @pytest.mark.skip(reason="Could not make this work") - def test_private_model_details_view(self): - response = self.client.get( - reverse( - "models:details_private", - kwargs={"user": self.user, "project": self.project.name, "id": self.private_model.id}, - ) - ) - - # Check status code - assert response.status_code == 200 - self.assertTemplateUsed(response, "models/models_details_private.html") - assert "Model Details" in response.content.decode() - assert "Private model test Details | SciLifeLab Serve (beta)" in response.content.decode() - - @pytest.mark.skip(reason="I am even not sure that it's being invoked") - def test_model_create_view(self): - response = self.client.get(reverse("models:create", kwargs={"user": self.user, "project": self.project.name})) - - # Check status code - assert response.status_code == 200 - self.assertTemplateUsed(response, "models/models_details_private.html") - assert "Model Details" in response.content.decode() - assert "Private model test Details | SciLifeLab Serve (beta)" in response.content.decode() - - @pytest.mark.skip(reason="I think that the project is not created for this") - def test_models_list_view(self): - response = self.client.get(reverse("models:list", kwargs={"user": self.user, "project": self.project.name})) - # Check status code - assert response.status_code == 200 - self.assertTemplateUsed(response, "models/models_list.html") - assert "Models" in response.content.decode() - assert f"{self.project.name} - Models | SciLifeLab Serve (beta)" in response.content.decode() - - -class ModelViewForbidden(TestCase): - def setUp(self): - user = User.objects.create_user(test_user["username"], test_user["email"], test_user["password"]) - - project = Project.objects.create_project(name="test-perm", owner=user, description="") - - new_model = Model( - uid="test_uid", - name="test", - bucket="", - description="model_description", - model_card="", - project=project, - access="PR", - ) - new_model.save() - - user = User.objects.create_user("member", "bar@test.com", "bar") - self.client.login(username="bar@test.com", password="bar") - - def test_forbidden_models_list(self): - """ - Test non-project member not allowed to access /models - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - response = self.client.get(reverse("models:list", kwargs={"user": owner, "project": project.slug})) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - assert "Forbidden | SciLifeLab Serve (beta)" in response.content.decode() - - def test_forbidden_models_create(self): - """ - Test non-project member not allowed to access /models/create - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "models:create", - kwargs={"user": owner, "project": project.slug}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_details_private(self): - """ - Test non-project member not allowed to access /models/ - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:details_private", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_delete(self): - """ - Test non-project member not allowed to access /models//delete - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:delete", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_publish(self): - """ - Test non-project member not allowed to access /models//publish - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:publish_model", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_add_tag(self): - """ - Test non-project member not allowed to access /models//add_tag - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:add_tag_private", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_remove_tag(self): - """ - Test non-project member not allowed to - access /models//remove_tag - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:remove_tag_private", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_unpublidh(self): - """ - Test non-project member not allowed to - access /models//unpublish - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:unpublish_model", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_access(self): - """ - Test non-project member not allowed to access /models//access - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:change_access", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_upload(self): - """ - Test non-project member not allowed to access /models//upload - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:upload_model_headline", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_models_docker(self): - """ - Test non-project member not allowed to access /models//docker - """ - owner = User.objects.get(username=test_user["email"]) - project = Project.objects.get(name="test-perm") - model = Model.objects.get(name="test") - response = self.client.get( - reverse( - "models:add_docker_image", - kwargs={ - "user": owner, - "project": project.slug, - "id": model.id, - }, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - -class TestFixtures(TestCase): - fixtures = ["models/fixtures/objecttype_fixtures.json"] - - def test_objecttype_mlflow(self): - obj_type = ObjectType.objects.get(slug="mlflow") - self.assertEqual(obj_type.slug, "mlflow") - - def test_objecttype_tfmodel(self): - obj_type = ObjectType.objects.get(slug="tensorflow") - self.assertEqual(obj_type.slug, "tensorflow") - - def test_objecttype_pythonmodel(self): - obj_type = ObjectType.objects.get(slug="python") - self.assertEqual(obj_type.slug, "python") - - def test_objecttype_defaultmodel(self): - obj_type = ObjectType.objects.get(slug="default") - self.assertEqual(obj_type.slug, "default") - - def test_objecttype_pytorchmodel(self): - obj_type = ObjectType.objects.get(slug="pytorch") - self.assertEqual(obj_type.slug, "pytorch") diff --git a/models/views.py b/models/views.py index bab5c04cc..bfb358f53 100644 --- a/models/views.py +++ b/models/views.py @@ -27,7 +27,7 @@ logger = logging.getLogger(__name__) Apps = apps.get_model(app_label=settings.APPS_MODEL) -AppInstance = apps.get_model(app_label=settings.APPINSTANCE_MODEL) +BaseAppInstance = apps.get_model(app_label="apps.BaseAppInstance") Project = apps.get_model(app_label=settings.PROJECTS_MODEL) ProjectLog = apps.get_model(app_label=settings.PROJECTLOG_MODEL) @@ -53,7 +53,7 @@ def get(self, request, user, project): # For showing the persistent volumes currently # available within the project volumeK8s_set = Apps.objects.get(slug="volumeK8s") - volumes = AppInstance.objects.filter(Q(app=volumeK8s_set), Q(state="Running")) + volumes = BaseAppInstance.objects.filter(Q(app=volumeK8s_set), Q(app_status__status="Running")) # Passing the current project to the view/template project = ( @@ -75,7 +75,7 @@ def get(self, request, user, project): # for the time being is hard-coded to # jupyter-lab where usually models are trained app_set = Apps.objects.get(slug="jupyter-lab") - apps = AppInstance.objects.filter(Q(app=app_set), Q(state="Running")) + apps = BaseAppInstance.objects.filter(Q(app=app_set), Q(app_status__status="Running")) return render(request, self.template, locals()) @@ -125,7 +125,9 @@ def post(self, request, user, project): # The minio sidecar does this. # First find the minio release name minio_set = Apps.objects.get(slug="minio") - minio = AppInstance.objects.filter(Q(app=minio_set), Q(project=model_project), Q(state="Running")).first() + minio = BaseAppInstance.objects.filter( + Q(app=minio_set), Q(project=model_project), Q(app_status__status="Running") + ).first() minio_release = minio.parameters["release"] # e.g 'rfc058c6f' # Now find the related pod diff --git a/monitor/.github/dependabot.yml b/monitor/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/monitor/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/monitor/.github/workflows/branch-name-check.yaml b/monitor/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 03c9b39bf..000000000 --- a/monitor/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v2 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/monitor/.github/workflows/code-checks.yaml b/monitor/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/monitor/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/monitor/.gitignore b/monitor/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/monitor/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/monitor/LICENSE b/monitor/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/monitor/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/monitor/MANIFEST.in b/monitor/MANIFEST.in deleted file mode 100644 index a40571a8b..000000000 --- a/monitor/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include monitor/templates * diff --git a/monitor/README.md b/monitor/README.md deleted file mode 100644 index fee3cf432..000000000 --- a/monitor/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-monitor - -Monitor Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "monitor" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/monitor/__init__.py b/monitor/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/monitor/admin.py b/monitor/admin.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/monitor/apps.py b/monitor/apps.py deleted file mode 100644 index e11e7ae7e..000000000 --- a/monitor/apps.py +++ /dev/null @@ -1,7 +0,0 @@ -from django.apps import AppConfig - - -class MonitorConfig(AppConfig): - default_auto_field = "django.db.models.BigAutoField" - name = "monitor" - verbose_name = "Studio Monitor" diff --git a/monitor/dash_demo.py b/monitor/dash_demo.py deleted file mode 100644 index 05e9b1572..000000000 --- a/monitor/dash_demo.py +++ /dev/null @@ -1,201 +0,0 @@ -import dash_bootstrap_components as dbc -import dash_core_components as dcc -import dash_html_components as html -import requests -from dash.dependencies import Input, Output -from dash.exceptions import PreventUpdate -from django_plotly_dash import DjangoDash - -from apps.models import AppInstance, Apps -from projects.models import Project - -app = DjangoDash("FEDnDashboard") - - -menu = html.Div( - [ - dbc.Row( - [ - dbc.Col( - [ - dbc.Card( - [ - dbc.CardBody( - [ - dbc.Select( - id="reducer-select", - options=[], - className="form-control", - ) - ] - ) - ] - ) - ], - className="col-4", - ), - dbc.Col(id="reducer-state", className="col-2"), - ] - ), - dbc.Row(id="submenu"), - ] -) - -main = dbc.Row( - [ - dbc.Col( - [ - dbc.Card( - [ - dbc.CardBody( - [ - html.H4("Combiners", className="card-title"), - dbc.Table( - [ - html.Thead( - [ - html.Tr( - [ - html.Th("Name"), - html.Th("Active Clients"), - html.Th("IP"), - html.Th("Country"), - html.Th("Region"), - html.Th("City"), - ] - ) - ] - ), - html.Tbody([], id="combiner-info-table"), - ] - ), - ] - ) - ] - ), - dbc.Card( - [ - dbc.CardBody( - [ - html.H4("Round Time", className="card-title"), - dcc.Graph(id="combiners-round-plot"), - ] - ) - ] - ), - dbc.Card( - [ - dbc.CardBody( - [ - html.H4("Combiner Load", className="card-title"), - dcc.Graph(id="combiners-combiner-plot"), - ] - ) - ] - ), - dbc.Card( - [ - dbc.CardBody( - [ - html.H4( - "Controller MEM and CPU Monitoring", - className="card-title", - ), - dcc.Graph(id="combiners-memcpu-plot"), - ] - ) - ] - ), - ] - ) - ] -) - -app.layout = html.Div([menu, main]) - - -@app.callback( - Output(component_id="combiner-info-table", component_property="children"), - Output(component_id="combiners-round-plot", component_property="figure"), - Output(component_id="combiners-combiner-plot", component_property="figure"), - Output(component_id="combiners-memcpu-plot", component_property="figure"), - Output(component_id="reducer-state", component_property="children"), - Output(component_id="reducer-select", component_property="options"), - Input(component_id="reducer-select", component_property="value"), -) -def reducer_select(red_select, request): - if "current_project" in request.session: - project_slug = request.session["current_project"] - else: - raise PreventUpdate() - reducer_app = Apps.objects.get(slug="reducer") - this_project = Project.objects.get(slug=project_slug) - reducers = AppInstance.objects.filter(app=reducer_app, project=this_project) - options = [] - for reducer in reducers: - options.append({"label": reducer.name, "value": str(reducer.pk)}) - - combiners_info = [] - roundplot = {} - combinerplot = {} - memcpuplot = {} - state = "" - if red_select is not None: - # TODO: Check that user has access to project. - sel_reducer = AppInstance.objects.get(pk=red_select, project__slug=project_slug) - reducer_params = sel_reducer.parameters - r_host = reducer_params["release"] - r_domain = reducer_params["global"]["domain"] - - try: - url = "https://{}.{}/api/state".format(r_host, r_domain) - res = requests.get(url, verify=False) - current_state = res.json()["state"] - except Exception as err: - print(err) - state = dbc.Card([dbc.CardBody(current_state)]) - - try: - url = "https://{}.{}/api/combiners/info".format(r_host, r_domain) - print(url) - res = requests.get(url, verify=False) - combiners_raw = res.json() - except Exception as err: - print(err) - - for combiner_raw in combiners_raw: - row = [ - html.Td(combiner_raw["name"]), - html.Td(combiner_raw["nr_active_clients"]), - html.Td(combiner_raw["ip"]), - html.Td(combiner_raw["country"]), - html.Td(combiner_raw["region"]), - html.Td(combiner_raw["city"]), - ] - combiners_info.append(html.Tr(row)) - - try: - url = "https://{}.{}/api/combiners/roundplot".format(r_host, r_domain) - print(url) - res = requests.get(url, verify=False) - roundplot = res.json() - except Exception as err: - print(err) - - try: - url = "https://{}.{}/api/combiners/combinerplot".format(r_host, r_domain) - print(url) - res = requests.get(url, verify=False) - combinerplot = res.json() - except Exception as err: - print(err) - - try: - url = "https://{}.{}/api/combiners/memcpuplot".format(r_host, r_domain) - print(url) - res = requests.get(url, verify=False) - memcpuplot = res.json() - except Exception as err: - print(err) - - return combiners_info, roundplot, combinerplot, memcpuplot, state, options diff --git a/monitor/helpers.py b/monitor/helpers.py deleted file mode 100644 index 6540dea26..000000000 --- a/monitor/helpers.py +++ /dev/null @@ -1,178 +0,0 @@ -import logging - -import requests as r -from django.conf import settings - - -def pod_up(app_name): - num_pods_up = 0 - num_pods_count = 0 - try: - query_up = 'sum(up{app="' + app_name + '"})' - query_count = 'count(up{app="' + app_name + '"})' - print(query_up) - print(query_count) - print(settings.PROMETHEUS_SVC + "/api/v1/query") - response = r.get( - settings.PROMETHEUS_SVC + "/api/v1/query", - params={"query": query_up}, - ) - num_pods_up = 0 - num_pods_count = 0 - if response: - print(response.text) - result_up_json = response.json() - result_up = result_up_json["data"]["result"] - - response = r.get( - settings.PROMETHEUS_SVC + "/api/v1/query", - params={"query": query_count}, - ) - if response: - result_count_json = response.json() - result_count = result_count_json["data"]["result"] - num_pods_up = result_up[0]["value"][1] - num_pods_count = result_count[0]["value"][1] - except Exception: - logging.warning("Failed to fetch status of app " + app_name) - pass - - return num_pods_up, num_pods_count - - -def get_count_over_time(name, app_name, path, status_code, time_span): - total_count = 0 - # Strip path of potential / - path = path.replace("/", "") - try: - query = ( - "sum(max_over_time(" - + name - + '{app="' - + app_name - + '", path="/' - + path - + '/", status_code="' - + status_code - + '"}[' - + time_span - + "]))" - ) - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - if response: - total_count = response.json()["data"]["result"][0]["value"][1] - except Exception: - print("Failed to get total count for: {}, {}, {}.".format(app_name, path, status_code)) - - return total_count - - -def get_total_labs_cpu_usage_60s(project_slug): - query = ( - 'sum(sum (rate (container_cpu_usage_seconds_total{image!=""}[60s])) by (pod) * on(pod) group_left kube_pod_labels{label_project="' # noqa: E501 - + project_slug - + '", label_app="lab"})' - ) - print(query) - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - if result: - cpu_usage = result[0]["value"][1] - return "{:.1f}".format(float(cpu_usage)) - return 0 - - -def get_total_cpu_usage_60s_ts(project_slug, resource_type): - query = ( - '''(sum (sum ( irate (container_cpu_usage_seconds_total{image!=""}[60s] ) ) by (pod) * on(pod) group_left kube_pod_labels{label_type="''' # noqa: E501 - + resource_type - + '",label_project="' - + project_slug - + '"})) [30m:30s]' - ) - print(query) - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - # print(response.json()) - result = response.json()["data"]["result"] - if result: - return result[0]["values"] - return 0 - - -def get_total_labs_memory_usage_60s(project_slug): - query = ( - 'sum(sum (rate (container_memory_usage_bytes{image!=""}[60s])) by (pod) * on(pod) group_left kube_pod_labels{label_project="' # noqa: E501 - + project_slug - + '", label_app="lab"})' - ) - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - if result: - memory_usage = result[0]["value"][1] - return "{:.3f}".format(float(memory_usage) / 1e9 * 0.931323) - return 0 - - -def get_labs_memory_requests(project_slug): - query = ( - ( - "sum(kube_pod_container_resource_requests_memory_bytes" - ' * on(pod) group_left kube_pod_labels{label_project="' - ) - + project_slug - + '"})' - ) - # query = 'kube_pod_container_resource_requests_cpu_cores' - - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - if result: - memory = result[0]["value"][1] - return "{:.2f}".format(float(memory) / 1e9 * 0.931323) - return 0 - - -def get_labs_cpu_requests(project_slug): - query = ( - 'sum(kube_pod_container_resource_requests_cpu_cores * on(pod) group_left kube_pod_labels{label_project="' # noqa: E501 - + project_slug - + '"})' - ) - - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - if result: - num_cpus = result[0]["value"][1] - return num_cpus - return 0 - - -def get_resource(project_slug, resource_type, q_type, mem_or_cpu, app_name=[]): - query = ( - "sum(kube_pod_container_resource_" - + q_type - + "_" - + mem_or_cpu - + ' * on(pod) group_left kube_pod_labels{label_project="' - + project_slug - + '", label_type="' - + resource_type - + '"' - ) - if app_name: - query += ', label_app="' + app_name + '"})' - else: - query += "})" - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - if result: - res = result[0]["value"][1] - return res - return "0.0" - - -def get_all(): - query = 'kube_pod_container_resource_limits_memory_bytes * on(pod) group_left kube_pod_labels{label_type="lab", label_project="stochss-dev-tiz"}' # noqa: E501 - response = r.get(settings.PROMETHEUS_SVC + "/api/v1/query", params={"query": query}) - result = response.json()["data"]["result"] - print(result) diff --git a/monitor/migrations/__init__.py b/monitor/migrations/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/monitor/models.py b/monitor/models.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/monitor/setup.py b/monitor/setup.py deleted file mode 100644 index 43a00a99e..000000000 --- a/monitor/setup.py +++ /dev/null @@ -1,27 +0,0 @@ -from setuptools import setup - -setup( - name="studio-monitor", - version="0.0.1", - description="""Django app for handling portal in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["monitor"], - package_dir={"monitor": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.1", - "requests==2.31.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/monitor/templates/monitor_new.html b/monitor/templates/monitor_new.html deleted file mode 100644 index 92282fdc5..000000000 --- a/monitor/templates/monitor_new.html +++ /dev/null @@ -1,33 +0,0 @@ -{% extends 'baseproject.html' %} -{% load static %} - -{% load plotly_dash %} - -{% block projects_dropdown %} - -{% endblock %} - -{% block content %} - -

FEDn Dashboard

- -{% plotly_direct name="FEDnDashboard" %} - -{% endblock %} diff --git a/monitor/templates/monitor_overview.html b/monitor/templates/monitor_overview.html deleted file mode 100644 index 291f0b326..000000000 --- a/monitor/templates/monitor_overview.html +++ /dev/null @@ -1,10 +0,0 @@ -{% extends 'baseproject.html' %} - -{% block content %} - -

Monitor

- -
- -
-{% endblock %} diff --git a/monitor/tests.py b/monitor/tests.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/monitor/urls.py b/monitor/urls.py deleted file mode 100644 index a659306e0..000000000 --- a/monitor/urls.py +++ /dev/null @@ -1,18 +0,0 @@ -from django.urls import path - -from . import views - -app_name = "monitor" - -urlpatterns = [ - path("monitor/", views.overview, name="overview"), - path("liveout", views.liveout, name="liveout"), - path("usage", views.usage, name="usage"), - path("/cpuchart", views.cpuchart, name="cpuchart"), - path("lab/delete/", views.delete_lab, name="delete_lab"), - path( - "serve/delete/", - views.delete_deployment, - name="delete_deployment", - ), -] diff --git a/monitor/views.py b/monitor/views.py deleted file mode 100644 index c3a2d7531..000000000 --- a/monitor/views.py +++ /dev/null @@ -1,267 +0,0 @@ -import itertools -import time -from datetime import datetime - -from django.conf import settings as sett -from django.contrib.auth.decorators import login_required -from django.db.models import F, Q, Sum -from django.http import HttpResponse, HttpResponseRedirect, JsonResponse -from django.shortcuts import render, reverse - -from apps.models import ResourceData -from models.models import Model -from projects.models import Project - -from .helpers import get_resource, get_total_cpu_usage_60s_ts - - -def get_cpu_mem(resources, project_slug, resource_type): - res_list = list() - for resource in resources: - res_cpu_limit = get_resource( - project_slug, - resource_type, - "limits", - "cpu_cores", - app_name=resource.appname, - ) - res_cpu_limit = "{:.2f}".format(float(res_cpu_limit)) - res_cpu_request = get_resource( - project_slug, - resource_type, - "requests", - "cpu_cores", - app_name=resource.appname, - ) - res_cpu_limit = "{:.2f}".format(float(res_cpu_request)) - res_mem_limit = get_resource( - project_slug, - resource_type, - "limits", - "memory_bytes", - app_name=resource.appname, - ) - res_mem_limit = "{:.2f}".format(float(res_mem_limit) / 1e9 * 0.931323) - res_mem_request = get_resource( - project_slug, - resource_type, - "requests", - "memory_bytes", - app_name=resource.appname, - ) - res_mem_request = "{:.2f}".format(float(res_mem_request) / 1e9 * 0.931323) - - if resource_type == "lab": - res_owner = resource.lab_session_owner.username - res_flavor = resource.flavor_slug - res_id = str(resource.id) - res_name = resource.name - res_project = resource.project.name - res_status = resource.status - res_created = resource.created_at - res_updated = resource.updated_at - - res_list.append( - ( - res_owner, - res_flavor, - res_cpu_limit, - res_cpu_request, - res_mem_limit, - res_mem_request, - res_id, - res_name, - res_project, - res_status, - res_created, - res_updated, - ) - ) - - elif resource_type == "deployment": - res_owner = resource.created_by - res_model = resource.model.name - res_version = resource.model.version - res_id = resource.model.id - res_project = resource.deployment.project.name - res_name = resource.deployment.name - res_access = resource.access - res_endpoint = resource.endpoint - res_created = resource.created_at - - res_list.append( - ( - res_owner, - res_cpu_limit, - res_cpu_request, - res_mem_limit, - res_mem_request, - res_id, - res_model, - res_version, - res_project, - res_name, - res_access, - res_endpoint, - res_created, - ) - ) - - return res_list - - -@login_required -def liveout(request, user, project): - is_authorized = True - user_permissions = get_permissions(request, project, sett.MONITOR_PERM) # noqa: F821 - - if not user_permissions["view"]: - request.session["oidc_id_token_expiration"] = -1 - request.session.save() - # return HttpResponse('Not authorized', status=401) - is_authorized = False - template = "monitor2.html" - project = Project.objects.filter(slug=project).first() - - return render(request, template, locals()) - - -@login_required -def overview(request, user, project): - try: - projects = Project.objects.filter(Q(owner=request.user) | Q(authorized=request.user), status="active").distinct( - "pk" - ) - except TypeError as err: - projects = [] - print(err) - - is_authorized = True - user_permissions = get_permissions(request, project, sett.MONITOR_PERM) # noqa: F821 - if not user_permissions["view"]: - request.session["oidc_id_token_expiration"] = -1 - request.session.save() - # return HttpResponse('Not authorized', status=401) - is_authorized = False - - request.session["current_project"] = project - template = "monitor_new.html" - project = Project.objects.filter(slug=project).first() - - # resource_types = ['lab', 'deployment'] - # q_types = ['requests', 'limits'] - # r_types = ['memory_bytes', 'cpu_cores'] - - # resource_status = dict() - # for resource_type in resource_types: - # resource_status[resource_type] = dict() - # for q_type in q_types: - # resource_status[resource_type][q_type] = dict() - # for r_type in r_types: - # tmp = get_resource(project.slug, resource_type, q_type, r_type) # noqa E501 - - # if r_type == 'memory_bytes': - # tmp ="{:.2f}".format(float(tmp)/1e9*0.931323) - # elif tmp: - # tmp = "{:.2f}".format(float(tmp)) - - # resource_status[resource_type][q_type][r_type] = tmp - - # total_cpu = float(resource_status['lab']['limits']['cpu_cores'])+float(resource_status['deployment']['limits']['cpu_cores']) # noqa E501 - # total_mem = float(resource_status['lab']['limits']['memory_bytes'])+float(resource_status['deployment']['limits']['memory_bytes']) # noqa E501 - # total_cpu_req = float(resource_status['lab']['requests']['cpu_cores'])+float(resource_status['deployment']['requests']['cpu_cores']) # noqa E501 - # total_mem_req = float(resource_status['lab']['requests']['memory_bytes'])+float(resource_status['deployment']['requests']['memory_bytes']) # noqa E501 - - # labs = Session.objects.filter(project=project) - # lab_list = get_cpu_mem(labs, project.slug, 'lab') - - # deps = DeploymentInstance.objects.filter(model__project=project) - # dep_list = get_cpu_mem(deps, project.slug, 'deployment') - - return render(request, template, locals()) - - -def delete_lab(request, user, project, uid): - # project = Project.objects.filter(Q(slug=project), Q(owner=request.user) | Q(authorized=request.user)).first() # noqa E501 - # session = Session.objects.filter(Q(id=id), Q(project=project), Q(lab_session_owner=request.user)).first() # noqa E501 - user_permissions = get_permissions(request, project, sett.MONITOR_PERM) # noqa F821 - if not user_permissions["view"]: - request.session["oidc_id_token_expiration"] = -1 - request.session.save() - return HttpResponse("Not authorized", status=401) - project = Project.objects.get(slug=project) - session = Session.objects.get(id=uid, project=project) # noqa F821 - if session: - session.helmchart.delete() - - return HttpResponseRedirect( - reverse( - "monitor:overview", - kwargs={"user": request.user, "project": str(project.slug)}, - ) - ) - - -def delete_deployment(request, user, project, model_id): - user_permissions = get_permissions(request, project, sett.MONITOR_PERM) # noqa F821 - if not user_permissions["view"]: - request.session["oidc_id_token_expiration"] = -1 - request.session.save() - return HttpResponse("Not authorized", status=401) - model = Model.objects.get(id=model_id) - instance = DeploymentInstance.objects.get(model=model) # noqa F821 - instance.helmchart.delete() - return HttpResponseRedirect( - reverse( - "monitor:overview", - kwargs={"user": request.user, "project": project}, - ) - ) - - -def cpuchart(request, user, project, resource_type): - # labels = ['a', 'b', 'c'] - # data = [1, 3, 2] - labels = [] - data = [] - test = get_total_cpu_usage_60s_ts(project, resource_type) - for value in test: - tod = datetime.fromtimestamp(value[0]).strftime("%H:%M") - labels.append(tod) - data.append(value[1]) - return JsonResponse( - data={ - "labels": labels, - "data": data, - } - ) - - -def usage(request, user, project): - curr_timestamp = time.time() - points = ResourceData.objects.filter( - time__gte=curr_timestamp - 2 * 3600, appinstance__project__slug=project - ).order_by("time") - all_cpus = list() - for point in points: - all_cpus.append(point.cpu) - total = points.annotate(timeP=F("time")).values("timeP").annotate(total_cpu=Sum("cpu"), total_mem=Sum("mem")) - - labels = list(total.values_list("timeP")) - labels = list(itertools.chain.from_iterable(labels)) - step = 1 - np = 200 - if len(labels) > np: - step = round(len(labels) / np) - labels = labels[::step] - x_data = list() - for label in labels: - x_data.append(datetime.fromtimestamp(label).strftime("%H:%M:%S")) - - total_mem = list(total.values_list("total_mem")) - total_mem = list(itertools.chain.from_iterable(total_mem))[::step] - - total_cpu = list(total.values_list("total_cpu")) - total_cpu = list(itertools.chain.from_iterable(total_cpu))[::step] - - return JsonResponse(data={"labels": x_data, "data_cpu": total_cpu, "data_mem": total_mem}) diff --git a/news/LICENSE b/news/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/news/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/news/MANIFEST.in b/news/MANIFEST.in deleted file mode 100644 index 9268b357d..000000000 --- a/news/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include news/templates * diff --git a/news/README.md b/news/README.md deleted file mode 100644 index 311cb3ccf..000000000 --- a/news/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-portal - -Portal Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "portal" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/news/__init__.py b/news/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/news/admin.py b/news/admin.py deleted file mode 100644 index 9900e37a7..000000000 --- a/news/admin.py +++ /dev/null @@ -1,5 +0,0 @@ -from django.contrib import admin - -from .models import NewsObject - -admin.site.register(NewsObject) diff --git a/news/apps.py b/news/apps.py deleted file mode 100644 index e491f8688..000000000 --- a/news/apps.py +++ /dev/null @@ -1,7 +0,0 @@ -from django.apps import AppConfig - - -class AppsConfig(AppConfig): - default_auto_field = "django.db.models.BigAutoField" - name = "news" - verbose_name = "Serve News" diff --git a/news/migrations/0001_initial.py b/news/migrations/0001_initial.py deleted file mode 100644 index a957fcf8b..000000000 --- a/news/migrations/0001_initial.py +++ /dev/null @@ -1,20 +0,0 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - initial = True - - dependencies = [] - - operations = [ - migrations.CreateModel( - name="NewsObject", - fields=[ - ("created_on", models.DateTimeField(auto_now_add=True)), - ("title", models.CharField(default="", max_length=60, primary_key=True, serialize=False)), - ("body", models.TextField(blank=True, default="", max_length=2024, null=True)), - ], - ), - ] diff --git a/news/migrations/__init__.py b/news/migrations/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/news/models.py b/news/models.py deleted file mode 100644 index d70edea06..000000000 --- a/news/models.py +++ /dev/null @@ -1,18 +0,0 @@ -import random - -from django.conf import settings -from django.db import models - - -class NewsObject(models.Model): - created_on = models.DateTimeField(auto_now_add=True) - title = models.CharField(max_length=60, default="", primary_key=True) - body = models.TextField(blank=True, null=True, default="", max_length=2024) - - @property - def news_body(self): - return self.body - - @property - def news_title(self): - return self.title diff --git a/news/setup.py b/news/setup.py deleted file mode 100644 index 8afbd8171..000000000 --- a/news/setup.py +++ /dev/null @@ -1,28 +0,0 @@ -from setuptools import setup - -setup( - name="studio-portal", - version="0.0.1", - description="""Django app for handling news in Serve""", - url="https://www.scilifelab.se", - include_package_data=True, - package=["portal"], - package_dir={"portal": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.5", - "requests==2.31.0", - "Pillow==9.4.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/news/tasks.py b/news/tasks.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/news/tests.py b/news/tests.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/news/urls.py b/news/urls.py deleted file mode 100644 index 01008bdea..000000000 --- a/news/urls.py +++ /dev/null @@ -1,11 +0,0 @@ -from django.urls import path - -from projects.views import IndexView as IndexView - -from . import views - -app_name = "news" - -urlpatterns = [ - path("news/", views.news, name="news"), -] diff --git a/news/views.py b/news/views.py deleted file mode 100644 index 81c310cd6..000000000 --- a/news/views.py +++ /dev/null @@ -1,14 +0,0 @@ -from typing import Any - -import markdown -from django.http import HttpRequest, HttpResponse -from django.shortcuts import render - -from .models import NewsObject - - -def news(request: HttpRequest) -> HttpResponse: - news_objects = NewsObject.objects.all().order_by("-created_on") - for news in news_objects: - news.body_html = markdown.markdown(news.body) - return render(request, "news/news.html", {"news_objects": news_objects}) diff --git a/poetry.lock b/poetry.lock index 885e2410f..e80b421b7 100644 --- a/poetry.lock +++ b/poetry.lock @@ -216,13 +216,13 @@ tests-no-zope = ["attrs[tests-mypy]", "cloudpickle", "hypothesis", "pympler", "p [[package]] name = "aws-sam-translator" -version = "1.88.0" +version = "1.89.0" description = "AWS SAM Translator is a library that transform SAM templates into AWS CloudFormation templates" optional = true python-versions = "!=4.0,<=4.0,>=3.8" files = [ - {file = "aws_sam_translator-1.88.0-py3-none-any.whl", hash = "sha256:aa93d498d8de3fb3d485c316155b1628144b823bbc176099a20de06df666fcac"}, - {file = "aws_sam_translator-1.88.0.tar.gz", hash = "sha256:e77c65f3488566122277accd44a0f1ec018e37403e0d5fe25120d96e537e91a7"}, + {file = "aws_sam_translator-1.89.0-py3-none-any.whl", hash = "sha256:843be1b5ca7634f700ad0c844a7e0dc42858f35da502e91691473eadd1731ded"}, + {file = "aws_sam_translator-1.89.0.tar.gz", hash = "sha256:fff1005d0b1f3cb511d0ac7e85f54af06afc9d9e433df013a2338d7a0168d174"}, ] [package.dependencies] @@ -786,6 +786,24 @@ tomli = {version = "*", optional = true, markers = "python_full_version <= \"3.1 [package.extras] toml = ["tomli"] +[[package]] +name = "crispy-bootstrap5" +version = "2024.2" +description = "Bootstrap5 template pack for django-crispy-forms" +optional = false +python-versions = ">=3.8" +files = [ + {file = "crispy-bootstrap5-2024.2.tar.gz", hash = "sha256:7d1fa40c6faf472e30e85c72551a3d2c9eedbf0abfff920683315e4e6f670f2b"}, + {file = "crispy_bootstrap5-2024.2-py3-none-any.whl", hash = "sha256:3867e320920a6ef156e94f9e0f06a80344c453e1b3bd96cd9dc0522ae9e9afb8"}, +] + +[package.dependencies] +django = ">=4.2" +django-crispy-forms = ">=2" + +[package.extras] +test = ["pytest", "pytest-django"] + [[package]] name = "cron-descriptor" version = "1.4.3" @@ -2367,13 +2385,13 @@ xmp = ["defusedxml"] [[package]] name = "platformdirs" -version = "4.2.1" +version = "4.2.2" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a `user data dir`." optional = true python-versions = ">=3.8" files = [ - {file = "platformdirs-4.2.1-py3-none-any.whl", hash = "sha256:17d5a1161b3fd67b390023cb2d3b026bbd40abde6fdb052dfbd3a29c3ba22ee1"}, - {file = "platformdirs-4.2.1.tar.gz", hash = "sha256:031cd18d4ec63ec53e82dceaac0417d218a6863f7745dfcc9efe7793b7039bdf"}, + {file = "platformdirs-4.2.2-py3-none-any.whl", hash = "sha256:2d7a1657e36a80ea911db832a8a6ece5ee53d8de21edd5cc5879af6530b1bfee"}, + {file = "platformdirs-4.2.2.tar.gz", hash = "sha256:38b7b51f512eed9e84a22788b4bce1de17c0adb134d6becb09836e37d8654cd3"}, ] [package.extras] @@ -3012,90 +3030,90 @@ rpds-py = ">=0.7.0" [[package]] name = "regex" -version = "2024.5.10" +version = "2024.5.15" description = "Alternative regular expression module, to replace re." optional = true python-versions = ">=3.8" files = [ - {file = "regex-2024.5.10-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:eda3dd46df535da787ffb9036b5140f941ecb91701717df91c9daf64cabef953"}, - {file = "regex-2024.5.10-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:1d5bd666466c8f00a06886ce1397ba8b12371c1f1c6d1bef11013e9e0a1464a8"}, - {file = "regex-2024.5.10-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:32e5f3b8e32918bfbdd12eca62e49ab3031125c454b507127ad6ecbd86e62fca"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:534efd2653ebc4f26fc0e47234e53bf0cb4715bb61f98c64d2774a278b58c846"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:193b7c6834a06f722f0ce1ba685efe80881de7c3de31415513862f601097648c"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:160ba087232c5c6e2a1e7ad08bd3a3f49b58c815be0504d8c8aacfb064491cd8"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:951be1eae7b47660412dc4938777a975ebc41936d64e28081bf2e584b47ec246"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d8a0f0ab5453e409586b11ebe91c672040bc804ca98d03a656825f7890cbdf88"}, - {file = "regex-2024.5.10-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9e6d4d6ae1827b2f8c7200aaf7501c37cf3f3896c86a6aaf2566448397c823dd"}, - {file = "regex-2024.5.10-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:161a206c8f3511e2f5fafc9142a2cc25d7fe9a1ec5ad9b4ad2496a7c33e1c5d2"}, - {file = "regex-2024.5.10-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:44b3267cea873684af022822195298501568ed44d542f9a2d9bebc0212e99069"}, - {file = "regex-2024.5.10-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:560278c9975694e1f0bc50da187abf2cdc1e4890739ea33df2bc4a85eeef143e"}, - {file = "regex-2024.5.10-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:70364a097437dd0a90b31cd77f09f7387ad9ac60ef57590971f43b7fca3082a5"}, - {file = "regex-2024.5.10-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:42be5de7cc8c1edac55db92d82b68dc8e683b204d6f5414c5a51997a323d7081"}, - {file = "regex-2024.5.10-cp310-cp310-win32.whl", hash = "sha256:9a8625849387b9d558d528e263ecc9c0fbde86cfa5c2f0eef43fff480ae24d71"}, - {file = "regex-2024.5.10-cp310-cp310-win_amd64.whl", hash = "sha256:903350bf44d7e4116b4d5898b30b15755d61dcd3161e3413a49c7db76f0bee5a"}, - {file = "regex-2024.5.10-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:bf9596cba92ce7b1fd32c7b07c6e3212c7eed0edc271757e48bfcd2b54646452"}, - {file = "regex-2024.5.10-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:45cc13d398b6359a7708986386f72bd156ae781c3e83a68a6d4cee5af04b1ce9"}, - {file = "regex-2024.5.10-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ad45f3bccfcb00868f2871dce02a755529838d2b86163ab8a246115e80cfb7d6"}, - {file = "regex-2024.5.10-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:33d19f0cde6838c81acffff25c7708e4adc7dd02896c9ec25c3939b1500a1778"}, - {file = "regex-2024.5.10-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0a9f89d7db5ef6bdf53e5cc8e6199a493d0f1374b3171796b464a74ebe8e508a"}, - {file = "regex-2024.5.10-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8c6c71cf92b09e5faa72ea2c68aa1f61c9ce11cb66fdc5069d712f4392ddfd00"}, - {file = "regex-2024.5.10-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7467ad8b0eac0b28e52679e972b9b234b3de0ea5cee12eb50091d2b68145fe36"}, - {file = "regex-2024.5.10-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bc0db93ad039fc2fe32ccd3dd0e0e70c4f3d6e37ae83f0a487e1aba939bd2fbd"}, - {file = "regex-2024.5.10-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:fa9335674d7c819674467c7b46154196c51efbaf5f5715187fd366814ba3fa39"}, - {file = "regex-2024.5.10-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:7dda3091838206969c2b286f9832dff41e2da545b99d1cfaea9ebd8584d02708"}, - {file = "regex-2024.5.10-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:504b5116e2bd1821efd815941edff7535e93372a098e156bb9dffde30264e798"}, - {file = "regex-2024.5.10-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:91b53dea84415e8115506cc62e441a2b54537359c63d856d73cb1abe05af4c9a"}, - {file = "regex-2024.5.10-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:1a3903128f9e17a500618e80c68165c78c741ebb17dd1a0b44575f92c3c68b02"}, - {file = "regex-2024.5.10-cp311-cp311-win32.whl", hash = "sha256:236cace6c1903effd647ed46ce6dd5d76d54985fc36dafc5256032886736c85d"}, - {file = "regex-2024.5.10-cp311-cp311-win_amd64.whl", hash = "sha256:12446827f43c7881decf2c126762e11425de5eb93b3b0d8b581344c16db7047a"}, - {file = "regex-2024.5.10-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:14905ed75c7a6edf423eb46c213ed3f4507c38115f1ed3c00f4ec9eafba50e58"}, - {file = "regex-2024.5.10-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:4fad420b14ae1970a1f322e8ae84a1d9d89375eb71e1b504060ab2d1bfe68f3c"}, - {file = "regex-2024.5.10-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c46a76a599fcbf95f98755275c5527304cc4f1bb69919434c1e15544d7052910"}, - {file = "regex-2024.5.10-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0faecb6d5779753a6066a3c7a0471a8d29fe25d9981ca9e552d6d1b8f8b6a594"}, - {file = "regex-2024.5.10-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:aab65121229c2ecdf4a31b793d99a6a0501225bd39b616e653c87b219ed34a49"}, - {file = "regex-2024.5.10-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:50e7e96a527488334379e05755b210b7da4a60fc5d6481938c1fa053e0c92184"}, - {file = "regex-2024.5.10-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba034c8db4b264ef1601eb33cd23d87c5013b8fb48b8161debe2e5d3bd9156b0"}, - {file = "regex-2024.5.10-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:031219782d97550c2098d9a68ce9e9eaefe67d2d81d8ff84c8354f9c009e720c"}, - {file = "regex-2024.5.10-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:62b5f7910b639f3c1d122d408421317c351e213ca39c964ad4121f27916631c6"}, - {file = "regex-2024.5.10-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:cd832bd9b6120d6074f39bdfbb3c80e416848b07ac72910f1c7f03131a6debc3"}, - {file = "regex-2024.5.10-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:e91b1976358e17197157b405cab408a5f4e33310cda211c49fc6da7cffd0b2f0"}, - {file = "regex-2024.5.10-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:571452362d552de508c37191b6abbbb660028b8b418e2d68c20779e0bc8eaaa8"}, - {file = "regex-2024.5.10-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:5253dcb0bfda7214523de58b002eb0090cb530d7c55993ce5f6d17faf953ece7"}, - {file = "regex-2024.5.10-cp312-cp312-win32.whl", hash = "sha256:2f30a5ab8902f93930dc6f627c4dd5da2703333287081c85cace0fc6e21c25af"}, - {file = "regex-2024.5.10-cp312-cp312-win_amd64.whl", hash = "sha256:3799e36d60a35162bb35b2246d8bb012192b7437dff807ef79c14e7352706306"}, - {file = "regex-2024.5.10-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:bbdc5db2c98ac2bf1971ffa1410c87ca7a15800415f788971e8ba8520fc0fda9"}, - {file = "regex-2024.5.10-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6ccdeef4584450b6f0bddd5135354908dacad95425fcb629fe36d13e48b60f32"}, - {file = "regex-2024.5.10-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:29d839829209f3c53f004e1de8c3113efce6d98029f044fa5cfee666253ee7e6"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0709ba544cf50bd5cb843df4b8bb6701bae2b70a8e88da9add8386cbca5c1385"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:972b49f2fe1047b9249c958ec4fa1bdd2cf8ce305dc19d27546d5a38e57732d8"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9cdbb1998da94607d5eec02566b9586f0e70d6438abf1b690261aac0edda7ab6"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bf7c8ee4861d9ef5b1120abb75846828c811f932d63311596ad25fa168053e00"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7d35d4cc9270944e95f9c88af757b0c9fc43f396917e143a5756608462c5223b"}, - {file = "regex-2024.5.10-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8722f72068b3e1156a4b2e1afde6810f1fc67155a9fa30a4b9d5b4bc46f18fb0"}, - {file = "regex-2024.5.10-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:696639a73ca78a380acfaa0a1f6dd8220616a99074c05bba9ba8bb916914b224"}, - {file = "regex-2024.5.10-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:ea057306ab469130167014b662643cfaed84651c792948891d003cf0039223a5"}, - {file = "regex-2024.5.10-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:b43b78f9386d3d932a6ce5af4b45f393d2e93693ee18dc4800d30a8909df700e"}, - {file = "regex-2024.5.10-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:c43395a3b7cc9862801a65c6994678484f186ce13c929abab44fb8a9e473a55a"}, - {file = "regex-2024.5.10-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:0bc94873ba11e34837bffd7e5006703abeffc4514e2f482022f46ce05bd25e67"}, - {file = "regex-2024.5.10-cp38-cp38-win32.whl", hash = "sha256:1118ba9def608250250f4b3e3f48c62f4562ba16ca58ede491b6e7554bfa09ff"}, - {file = "regex-2024.5.10-cp38-cp38-win_amd64.whl", hash = "sha256:458d68d34fb74b906709735c927c029e62f7d06437a98af1b5b6258025223210"}, - {file = "regex-2024.5.10-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:15e593386ec6331e0ab4ac0795b7593f02ab2f4b30a698beb89fbdc34f92386a"}, - {file = "regex-2024.5.10-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ca23b41355ba95929e9505ee04e55495726aa2282003ed9b012d86f857d3e49b"}, - {file = "regex-2024.5.10-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:2c8982ee19ccecabbaeac1ba687bfef085a6352a8c64f821ce2f43e6d76a9298"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7117cb7d6ac7f2e985f3d18aa8a1728864097da1a677ffa69e970ca215baebf1"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b66421f8878a0c82fc0c272a43e2121c8d4c67cb37429b764f0d5ad70b82993b"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:224a9269f133564109ce668213ef3cb32bc72ccf040b0b51c72a50e569e9dc9e"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ab98016541543692a37905871a5ffca59b16e08aacc3d7d10a27297b443f572d"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:51d27844763c273a122e08a3e86e7aefa54ee09fb672d96a645ece0454d8425e"}, - {file = "regex-2024.5.10-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:853cc36e756ff673bf984e9044ccc8fad60b95a748915dddeab9488aea974c73"}, - {file = "regex-2024.5.10-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4e7eaf9df15423d07b6050fb91f86c66307171b95ea53e2d87a7993b6d02c7f7"}, - {file = "regex-2024.5.10-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:169fd0acd7a259f58f417e492e93d0e15fc87592cd1e971c8c533ad5703b5830"}, - {file = "regex-2024.5.10-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:334b79ce9c08f26b4659a53f42892793948a613c46f1b583e985fd5a6bf1c149"}, - {file = "regex-2024.5.10-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:f03b1dbd4d9596dd84955bb40f7d885204d6aac0d56a919bb1e0ff2fb7e1735a"}, - {file = "regex-2024.5.10-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:cfa6d61a76c77610ba9274c1a90a453062bdf6887858afbe214d18ad41cf6bde"}, - {file = "regex-2024.5.10-cp39-cp39-win32.whl", hash = "sha256:249fbcee0a277c32a3ce36d8e36d50c27c968fdf969e0fbe342658d4e010fbc8"}, - {file = "regex-2024.5.10-cp39-cp39-win_amd64.whl", hash = "sha256:0ce56a923f4c01d7568811bfdffe156268c0a7aae8a94c902b92fe34c4bde785"}, - {file = "regex-2024.5.10.tar.gz", hash = "sha256:304e7e2418146ae4d0ef0e9ffa28f881f7874b45b4994cc2279b21b6e7ae50c8"}, + {file = "regex-2024.5.15-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a81e3cfbae20378d75185171587cbf756015ccb14840702944f014e0d93ea09f"}, + {file = "regex-2024.5.15-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:7b59138b219ffa8979013be7bc85bb60c6f7b7575df3d56dc1e403a438c7a3f6"}, + {file = "regex-2024.5.15-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a0bd000c6e266927cb7a1bc39d55be95c4b4f65c5be53e659537537e019232b1"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5eaa7ddaf517aa095fa8da0b5015c44d03da83f5bd49c87961e3c997daed0de7"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ba68168daedb2c0bab7fd7e00ced5ba90aebf91024dea3c88ad5063c2a562cca"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:6e8d717bca3a6e2064fc3a08df5cbe366369f4b052dcd21b7416e6d71620dca1"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1337b7dbef9b2f71121cdbf1e97e40de33ff114801263b275aafd75303bd62b5"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f9ebd0a36102fcad2f03696e8af4ae682793a5d30b46c647eaf280d6cfb32796"}, + {file = "regex-2024.5.15-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9efa1a32ad3a3ea112224897cdaeb6aa00381627f567179c0314f7b65d354c62"}, + {file = "regex-2024.5.15-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:1595f2d10dff3d805e054ebdc41c124753631b6a471b976963c7b28543cf13b0"}, + {file = "regex-2024.5.15-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:b802512f3e1f480f41ab5f2cfc0e2f761f08a1f41092d6718868082fc0d27143"}, + {file = "regex-2024.5.15-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:a0981022dccabca811e8171f913de05720590c915b033b7e601f35ce4ea7019f"}, + {file = "regex-2024.5.15-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:19068a6a79cf99a19ccefa44610491e9ca02c2be3305c7760d3831d38a467a6f"}, + {file = "regex-2024.5.15-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:1b5269484f6126eee5e687785e83c6b60aad7663dafe842b34691157e5083e53"}, + {file = "regex-2024.5.15-cp310-cp310-win32.whl", hash = "sha256:ada150c5adfa8fbcbf321c30c751dc67d2f12f15bd183ffe4ec7cde351d945b3"}, + {file = "regex-2024.5.15-cp310-cp310-win_amd64.whl", hash = "sha256:ac394ff680fc46b97487941f5e6ae49a9f30ea41c6c6804832063f14b2a5a145"}, + {file = "regex-2024.5.15-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:f5b1dff3ad008dccf18e652283f5e5339d70bf8ba7c98bf848ac33db10f7bc7a"}, + {file = "regex-2024.5.15-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c6a2b494a76983df8e3d3feea9b9ffdd558b247e60b92f877f93a1ff43d26656"}, + {file = "regex-2024.5.15-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a32b96f15c8ab2e7d27655969a23895eb799de3665fa94349f3b2fbfd547236f"}, + {file = "regex-2024.5.15-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:10002e86e6068d9e1c91eae8295ef690f02f913c57db120b58fdd35a6bb1af35"}, + {file = "regex-2024.5.15-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ec54d5afa89c19c6dd8541a133be51ee1017a38b412b1321ccb8d6ddbeb4cf7d"}, + {file = "regex-2024.5.15-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:10e4ce0dca9ae7a66e6089bb29355d4432caed736acae36fef0fdd7879f0b0cb"}, + {file = "regex-2024.5.15-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3e507ff1e74373c4d3038195fdd2af30d297b4f0950eeda6f515ae3d84a1770f"}, + {file = "regex-2024.5.15-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d1f059a4d795e646e1c37665b9d06062c62d0e8cc3c511fe01315973a6542e40"}, + {file = "regex-2024.5.15-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:0721931ad5fe0dda45d07f9820b90b2148ccdd8e45bb9e9b42a146cb4f695649"}, + {file = "regex-2024.5.15-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:833616ddc75ad595dee848ad984d067f2f31be645d603e4d158bba656bbf516c"}, + {file = "regex-2024.5.15-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:287eb7f54fc81546346207c533ad3c2c51a8d61075127d7f6d79aaf96cdee890"}, + {file = "regex-2024.5.15-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:19dfb1c504781a136a80ecd1fff9f16dddf5bb43cec6871778c8a907a085bb3d"}, + {file = "regex-2024.5.15-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:119af6e56dce35e8dfb5222573b50c89e5508d94d55713c75126b753f834de68"}, + {file = "regex-2024.5.15-cp311-cp311-win32.whl", hash = "sha256:1c1c174d6ec38d6c8a7504087358ce9213d4332f6293a94fbf5249992ba54efa"}, + {file = "regex-2024.5.15-cp311-cp311-win_amd64.whl", hash = "sha256:9e717956dcfd656f5055cc70996ee2cc82ac5149517fc8e1b60261b907740201"}, + {file = "regex-2024.5.15-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:632b01153e5248c134007209b5c6348a544ce96c46005d8456de1d552455b014"}, + {file = "regex-2024.5.15-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:e64198f6b856d48192bf921421fdd8ad8eb35e179086e99e99f711957ffedd6e"}, + {file = "regex-2024.5.15-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:68811ab14087b2f6e0fc0c2bae9ad689ea3584cad6917fc57be6a48bbd012c49"}, + {file = "regex-2024.5.15-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f8ec0c2fea1e886a19c3bee0cd19d862b3aa75dcdfb42ebe8ed30708df64687a"}, + {file = "regex-2024.5.15-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d0c0c0003c10f54a591d220997dd27d953cd9ccc1a7294b40a4be5312be8797b"}, + {file = "regex-2024.5.15-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2431b9e263af1953c55abbd3e2efca67ca80a3de8a0437cb58e2421f8184717a"}, + {file = "regex-2024.5.15-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a605586358893b483976cffc1723fb0f83e526e8f14c6e6614e75919d9862cf"}, + {file = "regex-2024.5.15-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:391d7f7f1e409d192dba8bcd42d3e4cf9e598f3979cdaed6ab11288da88cb9f2"}, + {file = "regex-2024.5.15-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:9ff11639a8d98969c863d4617595eb5425fd12f7c5ef6621a4b74b71ed8726d5"}, + {file = "regex-2024.5.15-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:4eee78a04e6c67e8391edd4dad3279828dd66ac4b79570ec998e2155d2e59fd5"}, + {file = "regex-2024.5.15-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:8fe45aa3f4aa57faabbc9cb46a93363edd6197cbc43523daea044e9ff2fea83e"}, + {file = "regex-2024.5.15-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:d0a3d8d6acf0c78a1fff0e210d224b821081330b8524e3e2bc5a68ef6ab5803d"}, + {file = "regex-2024.5.15-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:c486b4106066d502495b3025a0a7251bf37ea9540433940a23419461ab9f2a80"}, + {file = "regex-2024.5.15-cp312-cp312-win32.whl", hash = "sha256:c49e15eac7c149f3670b3e27f1f28a2c1ddeccd3a2812cba953e01be2ab9b5fe"}, + {file = "regex-2024.5.15-cp312-cp312-win_amd64.whl", hash = "sha256:673b5a6da4557b975c6c90198588181029c60793835ce02f497ea817ff647cb2"}, + {file = "regex-2024.5.15-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:87e2a9c29e672fc65523fb47a90d429b70ef72b901b4e4b1bd42387caf0d6835"}, + {file = "regex-2024.5.15-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c3bea0ba8b73b71b37ac833a7f3fd53825924165da6a924aec78c13032f20850"}, + {file = "regex-2024.5.15-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:bfc4f82cabe54f1e7f206fd3d30fda143f84a63fe7d64a81558d6e5f2e5aaba9"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e5bb9425fe881d578aeca0b2b4b3d314ec88738706f66f219c194d67179337cb"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:64c65783e96e563103d641760664125e91bd85d8e49566ee560ded4da0d3e704"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cf2430df4148b08fb4324b848672514b1385ae3807651f3567871f130a728cc3"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5397de3219a8b08ae9540c48f602996aa6b0b65d5a61683e233af8605c42b0f2"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:455705d34b4154a80ead722f4f185b04c4237e8e8e33f265cd0798d0e44825fa"}, + {file = "regex-2024.5.15-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b2b6f1b3bb6f640c1a92be3bbfbcb18657b125b99ecf141fb3310b5282c7d4ed"}, + {file = "regex-2024.5.15-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:3ad070b823ca5890cab606c940522d05d3d22395d432f4aaaf9d5b1653e47ced"}, + {file = "regex-2024.5.15-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:5b5467acbfc153847d5adb21e21e29847bcb5870e65c94c9206d20eb4e99a384"}, + {file = "regex-2024.5.15-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:e6662686aeb633ad65be2a42b4cb00178b3fbf7b91878f9446075c404ada552f"}, + {file = "regex-2024.5.15-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:2b4c884767504c0e2401babe8b5b7aea9148680d2e157fa28f01529d1f7fcf67"}, + {file = "regex-2024.5.15-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:3cd7874d57f13bf70078f1ff02b8b0aa48d5b9ed25fc48547516c6aba36f5741"}, + {file = "regex-2024.5.15-cp38-cp38-win32.whl", hash = "sha256:e4682f5ba31f475d58884045c1a97a860a007d44938c4c0895f41d64481edbc9"}, + {file = "regex-2024.5.15-cp38-cp38-win_amd64.whl", hash = "sha256:d99ceffa25ac45d150e30bd9ed14ec6039f2aad0ffa6bb87a5936f5782fc1569"}, + {file = "regex-2024.5.15-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:13cdaf31bed30a1e1c2453ef6015aa0983e1366fad2667657dbcac7b02f67133"}, + {file = "regex-2024.5.15-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:cac27dcaa821ca271855a32188aa61d12decb6fe45ffe3e722401fe61e323cd1"}, + {file = "regex-2024.5.15-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:7dbe2467273b875ea2de38ded4eba86cbcbc9a1a6d0aa11dcf7bd2e67859c435"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:64f18a9a3513a99c4bef0e3efd4c4a5b11228b48aa80743be822b71e132ae4f5"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d347a741ea871c2e278fde6c48f85136c96b8659b632fb57a7d1ce1872547600"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1878b8301ed011704aea4c806a3cadbd76f84dece1ec09cc9e4dc934cfa5d4da"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4babf07ad476aaf7830d77000874d7611704a7fcf68c9c2ad151f5d94ae4bfc4"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:35cb514e137cb3488bce23352af3e12fb0dbedd1ee6e60da053c69fb1b29cc6c"}, + {file = "regex-2024.5.15-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:cdd09d47c0b2efee9378679f8510ee6955d329424c659ab3c5e3a6edea696294"}, + {file = "regex-2024.5.15-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:72d7a99cd6b8f958e85fc6ca5b37c4303294954eac1376535b03c2a43eb72629"}, + {file = "regex-2024.5.15-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:a094801d379ab20c2135529948cb84d417a2169b9bdceda2a36f5f10977ebc16"}, + {file = "regex-2024.5.15-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:c0c18345010870e58238790a6779a1219b4d97bd2e77e1140e8ee5d14df071aa"}, + {file = "regex-2024.5.15-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:16093f563098448ff6b1fa68170e4acbef94e6b6a4e25e10eae8598bb1694b5d"}, + {file = "regex-2024.5.15-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:e38a7d4e8f633a33b4c7350fbd8bad3b70bf81439ac67ac38916c4a86b465456"}, + {file = "regex-2024.5.15-cp39-cp39-win32.whl", hash = "sha256:71a455a3c584a88f654b64feccc1e25876066c4f5ef26cd6dd711308aa538694"}, + {file = "regex-2024.5.15-cp39-cp39-win_amd64.whl", hash = "sha256:cab12877a9bdafde5500206d1020a584355a97884dfd388af3699e9137bf7388"}, + {file = "regex-2024.5.15.tar.gz", hash = "sha256:d3ee02d9e5f482cc8309134a91eeaacbdd2261ba111b0fef3748eeb4913e6a2c"}, ] [[package]] @@ -3890,4 +3908,4 @@ tests = ["hypothesis", "moto", "pytest", "pytest-cov", "pytest-django", "pytest- [metadata] lock-version = "2.0" python-versions = "^3.10.0" -content-hash = "302cd2db53228e09bfa677d5e4149fdfc4afb20ca005ffae17a4683a32f826ab" +content-hash = "023e0d6956823b9a32d65002cf16295657edc719acb2b2708e3fade7770deb0c" diff --git a/portal/.github/dependabot.yml b/portal/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/portal/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/portal/.github/workflows/branch-name-check.yaml b/portal/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 03c9b39bf..000000000 --- a/portal/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v2 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/portal/.github/workflows/code-checks.yaml b/portal/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/portal/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/portal/.gitignore b/portal/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/portal/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/portal/LICENSE b/portal/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/portal/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/portal/MANIFEST.in b/portal/MANIFEST.in deleted file mode 100644 index c00fc5198..000000000 --- a/portal/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include portal/templates * diff --git a/portal/README.md b/portal/README.md deleted file mode 100644 index 311cb3ccf..000000000 --- a/portal/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-portal - -Portal Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "portal" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/portal/admin.py b/portal/admin.py index 8b776fb01..4a3d217a5 100644 --- a/portal/admin.py +++ b/portal/admin.py @@ -1,6 +1,18 @@ from django.contrib import admin -from .models import PublicModelObject, PublishedModel +from .models import Collection, NewsObject, PublicModelObject, PublishedModel + +class CollectionAdmin(admin.ModelAdmin): + readonly_fields = ["connected_apps"] + + def connected_apps(self, obj): + apps = obj.app_instances.all() + app_list = ", ".join([app.name for app in apps]) + return app_list or "No apps connected" + + +admin.site.register(Collection, CollectionAdmin) +admin.site.register(NewsObject) admin.site.register(PublishedModel) admin.site.register(PublicModelObject) diff --git a/portal/migrations/0001_initial.py b/portal/migrations/0001_initial.py new file mode 100644 index 000000000..9aed1a321 --- /dev/null +++ b/portal/migrations/0001_initial.py @@ -0,0 +1,74 @@ +# Generated by Django 5.0.2 on 2024-05-27 07:28 + +import django.db.models.deletion +from django.conf import settings +from django.db import migrations, models + + +class Migration(migrations.Migration): + initial = True + + dependencies = [ + ("models", "0001_initial"), + ("projects", "0001_initial"), + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ] + + operations = [ + migrations.CreateModel( + name="NewsObject", + fields=[ + ("created_on", models.DateTimeField(auto_now_add=True)), + ("title", models.CharField(default="", max_length=60, primary_key=True, serialize=False)), + ("body", models.TextField(blank=True, default="", max_length=2024, null=True)), + ], + ), + migrations.CreateModel( + name="Collection", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("created_on", models.DateTimeField(auto_now_add=True)), + ("name", models.CharField(default="", max_length=200, unique=True)), + ("description", models.TextField(blank=True, default="", max_length=1000)), + ("website", models.URLField(blank=True)), + ("logo", models.ImageField(blank=True, null=True, upload_to="collections/logos/")), + ("slug", models.SlugField(blank=True, unique=True)), + ("zenodo_community_id", models.CharField(blank=True, max_length=200, null=True)), + ( + "maintainer", + models.ForeignKey( + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="collection_maintainer", + to=settings.AUTH_USER_MODEL, + ), + ), + ], + ), + migrations.CreateModel( + name="PublicModelObject", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("obj", models.FileField(upload_to="models/objects/")), + ("updated_on", models.DateTimeField(auto_now=True)), + ("created_on", models.DateTimeField(auto_now_add=True)), + ("model", models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to="models.model")), + ], + ), + migrations.CreateModel( + name="PublishedModel", + fields=[ + ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), + ("name", models.CharField(max_length=512)), + ("pattern", models.CharField(default="", max_length=255)), + ("updated_on", models.DateTimeField(auto_now=True)), + ("created_on", models.DateTimeField(auto_now_add=True)), + ( + "collections", + models.ManyToManyField(blank=True, related_name="published_models", to="portal.collection"), + ), + ("model_obj", models.ManyToManyField(to="portal.publicmodelobject")), + ("project", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="projects.project")), + ], + ), + ] diff --git a/portal/models.py b/portal/models.py index 29177c604..f8f2697fd 100644 --- a/portal/models.py +++ b/portal/models.py @@ -1,9 +1,11 @@ import random from django.conf import settings +from django.contrib.auth import get_user_model from django.db import models from django.db.models.signals import pre_save from django.dispatch import receiver +from django.utils.text import slugify class PublicModelObject(models.Model): @@ -20,7 +22,7 @@ class PublishedModel(models.Model): pattern = models.CharField(max_length=255, default="") updated_on = models.DateTimeField(auto_now=True) created_on = models.DateTimeField(auto_now_add=True) - collections = models.ManyToManyField("collections_module.Collection", related_name="published_models", blank=True) + collections = models.ManyToManyField("portal.Collection", related_name="published_models", blank=True) @property def model_description(self): @@ -57,3 +59,42 @@ def on_project_save(sender, instance, **kwargs): pattern = f"pattern-{randint}" instance.pattern = pattern + + +class NewsObject(models.Model): + created_on = models.DateTimeField(auto_now_add=True) + title = models.CharField(max_length=60, default="", primary_key=True) + body = models.TextField(blank=True, null=True, default="", max_length=2024) + + @property + def news_body(self): + return self.body + + @property + def news_title(self): + return self.title + + +class Collection(models.Model): + created_on = models.DateTimeField(auto_now_add=True) + maintainer = models.ForeignKey( + get_user_model(), + on_delete=models.CASCADE, + related_name="collection_maintainer", + null=True, + ) + name = models.CharField(max_length=200, unique=True, default="") + description = models.TextField(blank=True, default="", max_length=1000) + website = models.URLField(max_length=200, blank=True) + logo = models.ImageField(upload_to="collections/logos/", null=True, blank=True) + slug = models.SlugField(unique=True, blank=True) + zenodo_community_id = models.CharField(max_length=200, null=True, blank=True) + # repositories source would be another field + + def __str__(self): + return self.name + + def save(self, *args, **kwargs): + if not self.slug: + self.slug = slugify(self.name) + super(Collection, self).save(*args, **kwargs) diff --git a/portal/setup.py b/portal/setup.py deleted file mode 100644 index d22a48990..000000000 --- a/portal/setup.py +++ /dev/null @@ -1,28 +0,0 @@ -from setuptools import setup - -setup( - name="studio-portal", - version="0.0.1", - description="""Django app for handling portal in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["portal"], - package_dir={"portal": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.1", - "requests==2.31.0", - "Pillow==9.4.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/portal/urls.py b/portal/urls.py index 7c61e587f..1381effda 100644 --- a/portal/urls.py +++ b/portal/urls.py @@ -12,5 +12,8 @@ path("teaching/", views.teaching, name="teaching"), path("privacy/", views.privacy, name="privacy"), path("apps/", views.public_apps, name="apps"), + path("news/", views.news, name="news"), + path("collections/", views.index, name="collections_index"), + path("collections//", views.collection, name="collection"), path("", views.HomeViewDynamic.as_view(), name="home-dynamic"), ] diff --git a/portal/views.py b/portal/views.py index ada167a15..911874163 100644 --- a/portal/views.py +++ b/portal/views.py @@ -2,24 +2,26 @@ from django.apps import apps from django.conf import settings from django.db.models import Q -from django.shortcuts import redirect, render +from django.shortcuts import get_object_or_404, redirect, render from django.views.generic import View +from apps.models import BaseAppInstance, SocialMixin from studio.utils import get_logger +from .models import NewsObject + logger = get_logger(__name__) -AppInstance = apps.get_model(app_label=settings.APPINSTANCE_MODEL) + Project = apps.get_model(app_label=settings.PROJECTS_MODEL) PublishedModel = apps.get_model(app_label=settings.PUBLISHEDMODEL_MODEL) -NewsObject = apps.get_model(app_label="news.NewsObject") -Collection = apps.get_model(app_label="collections_module.Collection") +Collection = apps.get_model(app_label="portal.Collection") # TODO minor refactor # 1. Change id to app_id as it's anti-pattern to override language reserved function names # 2. add type annotations -def get_public_apps(request, id=0, get_all=True, collection=None): +def get_public_apps(request, app_id=0, get_all=True, collection=None): try: projects = Project.objects.filter( Q(owner=request.user) | Q(authorized=request.user), status="active" @@ -42,28 +44,41 @@ def get_public_apps(request, id=0, get_all=True, collection=None): # add app id to app_tags object if "app_id_add" in request.GET: num_tags = int(request.GET["tag_count"]) - id = int(request.GET["app_id_add"]) - request.session["app_tags"][str(id)] = num_tags + app_id = int(request.GET["app_id_add"]) + request.session["app_tags"][str(app_id)] = num_tags # remove app id from app_tags object if "app_id_remove" in request.GET: num_tags = int(request.GET["tag_count"]) - id = int(request.GET["app_id_remove"]) - if str(id) in request.session["app_tags"]: - request.session["app_tags"].pop(str(id)) + app_id = int(request.GET["app_id_remove"]) + if str(app_id) in request.session["app_tags"]: + request.session["app_tags"].pop(str(app_id)) # reset app_tags if Apps Tab on Sidebar pressed - if id == 0: + if app_id == 0: if "tf_add" not in request.GET and "tf_remove" not in request.GET: request.session["app_tags"] = {} + published_apps = [] + if collection: - published_apps = AppInstance.objects.filter( - ~Q(state="Deleted"), access="public", collections__slug=collection - ).order_by("-updated_on") + # TODO: TIDY THIS UP! + + for subclass in SocialMixin.__subclasses__(): + print(subclass, flush=True) + published_apps_qs = subclass.objects.filter( + ~Q(app_status__status="Deleted"), access="public", collections__slug=collection + ).order_by("-updated_on") + print(published_apps_qs, flush=True) + published_apps.extend([app for app in published_apps_qs]) + else: - published_apps = AppInstance.objects.filter(~Q(state="Deleted"), access="public").order_by("-updated_on") + for subclass in SocialMixin.__subclasses__(): + published_apps_qs = subclass.objects.filter(~Q(app_status__status="Deleted"), access="public").order_by( + "-updated_on" + ) + published_apps.extend([app for app in published_apps_qs]) - if published_apps.count() >= 3 and not get_all: + if len(published_apps) >= 3 and not get_all: published_apps = published_apps[:3] else: published_apps = published_apps @@ -71,19 +86,18 @@ def get_public_apps(request, id=0, get_all=True, collection=None): # Similar to GetStatusView() in apps.views for app in published_apps: try: - app.latest_status = app.status.latest().status_type - - app.status_group = "success" if app.latest_status in settings.APPS_STATUS_SUCCESS else "warning" + app.status_group = "success" if app.app_status.status in settings.APPS_STATUS_SUCCESS else "warning" except: # noqa E722 TODO refactor: Add exception app.latest_status = "unknown" app.status_group = "unknown" # Extract app config for use in Django templates for app in published_apps: - app.image = app.parameters.get("appconfig", {}).get("image", "Not available") - app.port = app.parameters.get("appconfig", {}).get("port", "Not available") - app.userid = app.parameters.get("appconfig", {}).get("userid", "Not available") - app.pvc = app.parameters.get("apps", {}).get("volumeK8s") or None + if getattr(app, "k8s_values", False): + app.image = app.k8s_values.get("appconfig", {}).get("image", "Not available") + app.port = app.k8s_values.get("appconfig", {}).get("port", "Not available") + app.userid = app.k8s_values.get("appconfig", {}).get("userid", "Not available") + app.pvc = app.k8s_values.get("apps", {}).get("volumeK8s") or None # create session object to store ids for tag seacrh if it does not exist if "app_tag_filters" not in request.session: @@ -114,8 +128,8 @@ def get_public_apps(request, id=0, get_all=True, collection=None): return published_apps, request -def public_apps(request, id=0): - published_apps, request = get_public_apps(request, id=id) +def public_apps(request, app_id=0): + published_apps, request = get_public_apps(request, app_id=app_id) template = "portal/apps.html" return render(request, template, locals()) @@ -123,8 +137,8 @@ def public_apps(request, id=0): class HomeView(View): template = "portal/home.html" - def get(self, request, id=0): - published_apps, request = get_public_apps(request, id=id, get_all=False) + def get(self, request, app_id=0): + published_apps, request = get_public_apps(request, app_id=app_id, get_all=False) published_models = PublishedModel.objects.all() news_objects = NewsObject.objects.all().order_by("-created_on") for news in news_objects: @@ -169,7 +183,7 @@ def get(self, request): if request.user.is_authenticated: return redirect("projects/") else: - return HomeView.as_view()(request, id=0) + return HomeView.as_view()(request, app_id=0) def about(request): @@ -185,3 +199,36 @@ def teaching(request): def privacy(request): template = "portal/privacy.html" return render(request, template, locals()) + + +def news(request): + news_objects = NewsObject.objects.all().order_by("-created_on") + for news in news_objects: + news.body_html = markdown.markdown(news.body) + return render(request, "news/news.html", {"news_objects": news_objects}) + + +def index(request): + template = "collections/index.html" + + collection_objects = Collection.objects.all().order_by("-created_on") + + context = {"collection_objects": collection_objects} + + return render(request, template, context=context) + + +def collection(request, slug, app_id=0): + template = "collections/collection.html" + + collection = get_object_or_404(Collection, slug=slug) + collection_published_apps, request = get_public_apps(request, app_id=app_id, collection=slug) + collection_published_models = PublishedModel.objects.all().filter(collections__slug=slug) + + context = { + "collection": collection, + "collection_published_apps": collection_published_apps, + "collection_published_models": collection_published_models, + } + + return render(request, template, context=context) diff --git a/projects/.github/dependabot.yml b/projects/.github/dependabot.yml deleted file mode 100644 index 9e5479c22..000000000 --- a/projects/.github/dependabot.yml +++ /dev/null @@ -1,22 +0,0 @@ -version: 2 -updates: - - # Maintain dependencies for GitHub Actions - - package-ecosystem: "github-actions" - directory: "/" - schedule: - interval: "weekly" - target-branch: "develop" - labels: - - "github-actions dependencies" - - - package-ecosystem: "pip" - directory: "/" - schedule: - interval: "weekly" - # Raise pull requests for version updates - # to pip against the `develop` branch - target-branch: "develop" - # Labels on pull requests for version updates only - labels: - - "pip dependencies" diff --git a/projects/.github/workflows/branch-name-check.yaml b/projects/.github/workflows/branch-name-check.yaml deleted file mode 100644 index 03c9b39bf..000000000 --- a/projects/.github/workflows/branch-name-check.yaml +++ /dev/null @@ -1,21 +0,0 @@ -name: "branch name check" - -on: - push: - branches-ignore: - - develop - - main - -env: - BRANCH_REGEX: '^((feature|hotfix|bug|docs|dependabot|fix)\/.+)|(release\/v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?))$' - -jobs: - branch-name-check: - runs-on: ubuntu-20.04 - steps: - - name: checkout - uses: actions/checkout@v2 - - - name: branch name check - run: | - git rev-parse --abbrev-ref HEAD | grep -P "$BRANCH_REGEX" diff --git a/projects/.github/workflows/code-checks.yaml b/projects/.github/workflows/code-checks.yaml deleted file mode 100644 index 68676755a..000000000 --- a/projects/.github/workflows/code-checks.yaml +++ /dev/null @@ -1,52 +0,0 @@ -name: Code checks - -on: - push: - branches: - - main - - develop - pull_request: - branches: - - main - - develop - release: - types: [published] - -jobs: - check-code: - - runs-on: ubuntu-20.04 - services: - postgres: - image: postgres - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: postgres - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 - - - steps: - - uses: actions/checkout@v3 - - name: Set up Python - uses: actions/setup-python@v4 - with: - #Python version should be the same as the base image in Dockerfile - python-version: "3.8.10" - - name: Install dependencies - - run: | - python -m pip install --upgrade pip - pip install black==22.10.0 isort flake8 - if [ -f requirements.txt ]; then pip install --no-cache-dir -r requirements.txt; fi - - name: Check Python imports - run: | - isort . --check --diff --skip migrations --skip .venv --profile black -p studio -p projects -p models -p apps -p portal --line-length 79 - - name: Check Python formatting - run: | - black . --check --diff --line-length 79 --exclude migrations - - name: Check Python linting - run: | - flake8 . --exclude migrations diff --git a/projects/.gitignore b/projects/.gitignore deleted file mode 100644 index b6e47617d..000000000 --- a/projects/.gitignore +++ /dev/null @@ -1,129 +0,0 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -pip-wheel-metadata/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -.python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ diff --git a/projects/LICENSE b/projects/LICENSE deleted file mode 100644 index 261eeb9e9..000000000 --- a/projects/LICENSE +++ /dev/null @@ -1,201 +0,0 @@ - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/projects/MANIFEST.in b/projects/MANIFEST.in deleted file mode 100644 index a876fcedd..000000000 --- a/projects/MANIFEST.in +++ /dev/null @@ -1,2 +0,0 @@ -include README.md -recursive-include projects/templates * diff --git a/projects/README.md b/projects/README.md deleted file mode 100644 index 97d76674f..000000000 --- a/projects/README.md +++ /dev/null @@ -1,18 +0,0 @@ -# studio-projects - -Projects Django module for [Studio/STACKn](https://github.com/scaleoutsystems/stackn). To include source package in a Django project: - -``` -$ python3 -m venv .venv -$ source .venv/bin/activate -$ pip install . -``` -And add to installed apps in settings.py: - -``` -INSTALLED_APPS=[ - "projects" -] -``` - -For a complete project follow the link above and navigate to settings.py diff --git a/projects/admin.py b/projects/admin.py index 0485ccb20..5d0d8301d 100644 --- a/projects/admin.py +++ b/projects/admin.py @@ -1,27 +1,12 @@ from django.conf import settings from django.contrib import admin -from .models import ( - S3, - BasicAuth, - Environment, - Flavor, - MLFlow, - Project, - ProjectLog, - ProjectTemplate, - ReleaseName, -) +from .models import BasicAuth, Environment, Flavor, Project, ProjectLog, ProjectTemplate admin.site.register(ProjectTemplate) -# admin.site.register(Project) admin.site.register(Environment) -# admin.site.register(Flavor) admin.site.register(ProjectLog) -admin.site.register(S3) -admin.site.register(MLFlow) admin.site.register(BasicAuth) -admin.site.register(ReleaseName) @admin.register(Project) diff --git a/projects/apps.py b/projects/apps.py index f1a230364..6a9b6f278 100644 --- a/projects/apps.py +++ b/projects/apps.py @@ -4,4 +4,4 @@ class ProjectConfig(AppConfig): default_auto_field = "django.db.models.BigAutoField" name = "projects" - verbose_name = "Scaleout Projects" + verbose_name = "Serve Projects" diff --git a/projects/helpers.py b/projects/helpers.py deleted file mode 100644 index 17796dcef..000000000 --- a/projects/helpers.py +++ /dev/null @@ -1,25 +0,0 @@ -import base64 -import re - - -def urlify(s): - # Remove all non-word characters (everything except numbers and letters) - s = re.sub(r"[^\w\s]", "", s) - - # Replace all runs of whitespace with a single dash - s = re.sub(r"\s+", "-", s) - - return s - - -def get_minio_keys(project): - return { - "project_key": decrypt_key(project.project_key), - "project_secret": decrypt_key(project.project_secret), - } - - -def decrypt_key(key): - base64_bytes = key.encode("ascii") - result = base64.b64decode(base64_bytes) - return result.decode("ascii") diff --git a/studio/migrations/projects/0001_initial.py b/projects/migrations/0001_initial.py similarity index 56% rename from studio/migrations/projects/0001_initial.py rename to projects/migrations/0001_initial.py index 9de409fe9..fe0ff20f2 100644 --- a/studio/migrations/projects/0001_initial.py +++ b/projects/migrations/0001_initial.py @@ -1,4 +1,4 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 +# Generated by Django 5.0.2 on 2024-05-27 07:28 import django.db.models.deletion from django.conf import settings @@ -16,50 +16,6 @@ class Migration(migrations.Migration): ] operations = [ - migrations.CreateModel( - name="BasicAuth", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("name", models.CharField(max_length=512)), - ("password", models.CharField(blank=True, default="", max_length=100)), - ("username", models.CharField(blank=True, default="", max_length=100)), - ( - "owner", - models.ForeignKey(on_delete=django.db.models.deletion.DO_NOTHING, to=settings.AUTH_USER_MODEL), - ), - ], - ), - migrations.CreateModel( - name="MLFlow", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("created_at", models.DateTimeField(auto_now_add=True)), - ("mlflow_url", models.CharField(max_length=512)), - ("host", models.CharField(blank=True, default="", max_length=512)), - ("name", models.CharField(max_length=512)), - ("updated_at", models.DateTimeField(auto_now=True)), - ( - "app", - models.OneToOneField( - blank=True, - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="mlflowobj", - to="apps.appinstance", - ), - ), - ( - "basic_auth", - models.ForeignKey( - blank=True, null=True, on_delete=django.db.models.deletion.DO_NOTHING, to="projects.basicauth" - ), - ), - ( - "owner", - models.ForeignKey(on_delete=django.db.models.deletion.DO_NOTHING, to=settings.AUTH_USER_MODEL), - ), - ], - ), migrations.CreateModel( name="Project", fields=[ @@ -80,16 +36,6 @@ class Migration(migrations.Migration): ("project_key", models.CharField(max_length=512)), ("project_secret", models.CharField(max_length=512)), ("authorized", models.ManyToManyField(blank=True, to=settings.AUTH_USER_MODEL)), - ( - "mlflow", - models.OneToOneField( - blank=True, - null=True, - on_delete=django.db.models.deletion.SET_NULL, - related_name="project_mlflow", - to="projects.mlflow", - ), - ), ( "owner", models.ForeignKey( @@ -104,72 +50,67 @@ class Migration(migrations.Migration): }, ), migrations.CreateModel( - name="S3", + name="Flavor", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("access_key", models.CharField(max_length=512)), ("created_at", models.DateTimeField(auto_now_add=True)), - ("host", models.CharField(max_length=512)), + ("cpu_lim", models.TextField(blank=True, default="1000m", null=True)), + ("gpu_lim", models.TextField(blank=True, default="0", null=True)), + ("ephmem_lim", models.TextField(blank=True, default="200Mi", null=True)), + ("mem_lim", models.TextField(blank=True, default="3Gi", null=True)), + ("cpu_req", models.TextField(blank=True, default="200m", null=True)), + ("gpu_req", models.TextField(blank=True, default="0", null=True)), + ("ephmem_req", models.TextField(blank=True, default="200Mi", null=True)), + ("mem_req", models.TextField(blank=True, default="0.5Gi", null=True)), ("name", models.CharField(max_length=512)), - ("region", models.CharField(blank=True, default="", max_length=512)), - ("secret_key", models.CharField(max_length=512)), ("updated_at", models.DateTimeField(auto_now=True)), - ( - "app", - models.OneToOneField( - blank=True, - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="s3obj", - to="apps.appinstance", - ), - ), - ( - "owner", - models.ForeignKey(on_delete=django.db.models.deletion.DO_NOTHING, to=settings.AUTH_USER_MODEL), - ), ( "project", - models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="s3_project", to="projects.project" - ), + models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project"), ), ], ), migrations.CreateModel( - name="ReleaseName", + name="Environment", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), ("created_at", models.DateTimeField(auto_now_add=True)), - ("name", models.CharField(max_length=512)), - ("status", models.CharField(max_length=10)), + ("image", models.CharField(max_length=100)), + ("name", models.CharField(max_length=100)), + ("repository", models.CharField(blank=True, max_length=100, null=True)), + ("slug", models.CharField(blank=True, max_length=100, null=True)), + ("updated_at", models.DateTimeField(auto_now=True)), + ("public", models.BooleanField(default=False)), + ("app", models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="apps.apps")), ( - "app", + "project", models.ForeignKey( - blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to="apps.appinstance" + blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project" ), ), - ( - "project", - models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project"), - ), ], ), migrations.CreateModel( - name="ProjectTemplate", + name="BasicAuth", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("description", models.TextField(blank=True, null=True)), - ("image", models.ImageField(blank=True, null=True, upload_to="projecttemplates/images/")), ("name", models.CharField(max_length=512)), - ("revision", models.IntegerField(default=1)), - ("slug", models.CharField(default="", max_length=512)), - ("template", models.TextField(blank=True, null=True)), - ("enabled", models.BooleanField(default=True)), + ("password", models.CharField(blank=True, default="", max_length=100)), + ("username", models.CharField(blank=True, default="", max_length=100)), + ( + "owner", + models.ForeignKey(on_delete=django.db.models.deletion.DO_NOTHING, to=settings.AUTH_USER_MODEL), + ), + ( + "project", + models.ForeignKey( + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="ba_project", + to="projects.project", + ), + ), ], - options={ - "unique_together": {("slug", "revision")}, - }, ), migrations.CreateModel( name="ProjectLog", @@ -196,97 +137,28 @@ class Migration(migrations.Migration): ("project", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="projects.project")), ], ), - migrations.AddField( - model_name="project", - name="s3storage", - field=models.OneToOneField( - blank=True, - null=True, - on_delete=django.db.models.deletion.SET_NULL, - related_name="project_s3", - to="projects.s3", - ), - ), - migrations.AddField( - model_name="mlflow", - name="project", - field=models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="mlflow_project", to="projects.project" - ), - ), - migrations.AddField( - model_name="mlflow", - name="s3", - field=models.ForeignKey( - blank=True, null=True, on_delete=django.db.models.deletion.DO_NOTHING, to="projects.s3" - ), - ), migrations.CreateModel( - name="Flavor", + name="ProjectTemplate", fields=[ ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("created_at", models.DateTimeField(auto_now_add=True)), - ("cpu_lim", models.TextField(blank=True, default="1000m", null=True)), - ("gpu_lim", models.TextField(blank=True, default="0", null=True)), - ("ephmem_lim", models.TextField(blank=True, default="200Mi", null=True)), - ("mem_lim", models.TextField(blank=True, default="3Gi", null=True)), - ("cpu_req", models.TextField(blank=True, default="200m", null=True)), - ("gpu_req", models.TextField(blank=True, default="0", null=True)), - ("ephmem_req", models.TextField(blank=True, default="200Mi", null=True)), - ("mem_req", models.TextField(blank=True, default="0.5Gi", null=True)), + ("description", models.TextField(blank=True, null=True)), + ("image", models.ImageField(blank=True, null=True, upload_to="projecttemplates/images/")), ("name", models.CharField(max_length=512)), - ("updated_at", models.DateTimeField(auto_now=True)), - ( - "project", - models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project"), - ), - ], - ), - migrations.CreateModel( - name="Environment", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("created_at", models.DateTimeField(auto_now_add=True)), - ("image", models.CharField(max_length=100)), - ("name", models.CharField(max_length=100)), - ("repository", models.CharField(blank=True, max_length=100, null=True)), - ("slug", models.CharField(blank=True, max_length=100, null=True)), - ("updated_at", models.DateTimeField(auto_now=True)), - ("public", models.BooleanField(default=False)), - ("app", models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to="apps.apps")), - ( - "appenv", - models.ForeignKey( - blank=True, - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="envobj", - to="apps.appinstance", - ), - ), - ( - "project", - models.ForeignKey( - blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to="projects.project" - ), - ), - ( - "registry", - models.ForeignKey( - blank=True, - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="environments", - to="apps.appinstance", - ), - ), + ("revision", models.IntegerField(default=1)), + ("slug", models.CharField(default="", max_length=512)), + ("template", models.TextField(blank=True, null=True)), + ("enabled", models.BooleanField(default=True)), + ("available_apps", models.ManyToManyField(blank=True, related_name="available_apps", to="apps.apps")), ], + options={ + "unique_together": {("slug", "revision")}, + }, ), migrations.AddField( - model_name="basicauth", - name="project", + model_name="project", + name="project_template", field=models.ForeignKey( - null=True, on_delete=django.db.models.deletion.CASCADE, related_name="ba_project", to="projects.project" + null=True, on_delete=django.db.models.deletion.DO_NOTHING, to="projects.projecttemplate" ), ), ] diff --git a/projects/models.py b/projects/models.py index 1ccd0ca65..eab020f98 100644 --- a/projects/models.py +++ b/projects/models.py @@ -35,13 +35,7 @@ class BasicAuth(models.Model): class Environment(models.Model): app = models.ForeignKey(settings.APPS_MODEL, on_delete=models.CASCADE, null=True) - appenv = models.ForeignKey( - settings.APPINSTANCE_MODEL, - related_name="envobj", - null=True, - blank=True, - on_delete=models.CASCADE, - ) + created_at = models.DateTimeField(auto_now_add=True) image = models.CharField(max_length=100) name = models.CharField(max_length=100) @@ -51,13 +45,7 @@ class Environment(models.Model): null=True, blank=True, ) - registry = models.ForeignKey( - settings.APPINSTANCE_MODEL, - related_name="environments", - null=True, - blank=True, - on_delete=models.CASCADE, - ) + repository = models.CharField(max_length=100, blank=True, null=True) slug = models.CharField(max_length=100, null=True, blank=True) updated_at = models.DateTimeField(auto_now=True) @@ -85,68 +73,26 @@ class Flavor(models.Model): def __str__(self): return str(self.name) + def to_dict(self): + flavor_dict = dict( + flavor=dict( + requests={ + "cpu": self.cpu_req, + "memory": self.mem_req, + "ephemeral-storage": self.ephmem_req, + }, + limits={ + "cpu": self.cpu_lim, + "memory": self.mem_lim, + "ephemeral-storage": self.ephmem_lim, + }, + ) + ) + if self.gpu_req and int(self.gpu_req) > 0: + flavor_dict["flavor"]["requests"]["nvidia.com/gpu"] = (self.gpu_req,) + flavor_dict["flavor"]["limits"]["nvidia.com/gpu"] = self.gpu_lim -class S3(models.Model): - access_key = models.CharField(max_length=512) - app = models.OneToOneField( - settings.APPINSTANCE_MODEL, - on_delete=models.CASCADE, - null=True, - blank=True, - related_name="s3obj", - ) - created_at = models.DateTimeField(auto_now_add=True) - host = models.CharField(max_length=512) - name = models.CharField(max_length=512) - owner = models.ForeignKey(get_user_model(), on_delete=models.DO_NOTHING) - project = models.ForeignKey( - settings.PROJECTS_MODEL, - on_delete=models.CASCADE, - related_name="s3_project", - ) - region = models.CharField(max_length=512, blank=True, default="") - secret_key = models.CharField(max_length=512) - updated_at = models.DateTimeField(auto_now=True) - - def __str__(self): - return "{} ({})".format(self.name, self.project.slug) - - -class MLFlow(models.Model): - app = models.OneToOneField( - settings.APPINSTANCE_MODEL, - on_delete=models.CASCADE, - null=True, - blank=True, - related_name="mlflowobj", - ) - basic_auth = models.ForeignKey(BasicAuth, on_delete=models.DO_NOTHING, null=True, blank=True) - created_at = models.DateTimeField(auto_now_add=True) - mlflow_url = models.CharField(max_length=512) - host = models.CharField(max_length=512, blank=True, default="") - name = models.CharField(max_length=512) - owner = models.ForeignKey(get_user_model(), on_delete=models.DO_NOTHING) - project = models.ForeignKey( - settings.PROJECTS_MODEL, - on_delete=models.CASCADE, - related_name="mlflow_project", - ) - s3 = models.ForeignKey(S3, on_delete=models.DO_NOTHING, blank=True, null=True) - updated_at = models.DateTimeField(auto_now=True) - - def __str__(self): - return "{} ({})".format(self.name, self.project.slug) - - -"""Post save signal when creating an mlflow object""" - - -@receiver(post_save, sender=MLFlow) -def create_mlflow(sender, instance, created, **kwargs): - if created: - if instance.project and not instance.project.mlflow: - instance.project.mlflow = instance - instance.project.save() + return flavor_dict # it will become the default objects attribute for a Project model @@ -240,6 +186,7 @@ class ProjectTemplate(models.Model): revision = models.IntegerField(default=1) slug = models.CharField(max_length=512, default="") template = models.TextField(null=True, blank=True) + available_apps = models.ManyToManyField("apps.Apps", blank=True, related_name="available_apps") enabled = models.BooleanField(default=True) @@ -258,13 +205,7 @@ class Project(models.Model): created_at = models.DateTimeField(auto_now_add=True) clone_url = models.CharField(max_length=512, null=True, blank=True) description = models.TextField(null=True, blank=True) - mlflow = models.OneToOneField( - MLFlow, - on_delete=models.SET_NULL, - null=True, - blank=True, - related_name="project_mlflow", - ) + name = models.CharField(max_length=512) objects = ProjectManager() owner = models.ForeignKey(get_user_model(), on_delete=models.DO_NOTHING, related_name="owner") @@ -273,13 +214,6 @@ class Project(models.Model): pattern = models.CharField(max_length=255, default="") - s3storage = models.OneToOneField( - S3, - on_delete=models.SET_NULL, - null=True, - blank=True, - related_name="project_s3", - ) slug = models.CharField(max_length=512, unique=True) status = models.CharField(max_length=20, null=True, blank=True, default="active") updated_at = models.DateTimeField(auto_now=True) @@ -364,19 +298,3 @@ class ProjectLog(models.Model): headline = models.CharField(max_length=256) module = models.CharField(max_length=2, choices=MODULE_CHOICES, default="UN") project = models.ForeignKey(settings.PROJECTS_MODEL, on_delete=models.CASCADE) - - -class ReleaseName(models.Model): - app = models.ForeignKey( - settings.APPINSTANCE_MODEL, - on_delete=models.CASCADE, - null=True, - blank=True, - ) - created_at = models.DateTimeField(auto_now_add=True) - name = models.CharField(max_length=512) - status = models.CharField(max_length=10) - project = models.ForeignKey(settings.PROJECTS_MODEL, on_delete=models.CASCADE, null=True) - - def __str__(self): - return "{}-{}-{}".format(self.name, self.project, self.app) diff --git a/projects/setup.py b/projects/setup.py deleted file mode 100644 index b46dded86..000000000 --- a/projects/setup.py +++ /dev/null @@ -1,30 +0,0 @@ -from setuptools import setup - -setup( - name="studio-projects", - version="0.0.1", - description="""Django app for handling portal in Studio""", - url="https://www.scaleoutsystems.com", - include_package_data=True, - package=["projects"], - package_dir={"projects": "."}, - python_requires=">=3.6,<4", - install_requires=[ - "django==4.2.1", - "requests==2.31.0", - "django-guardian==2.4.0", - "celery==5.2.7", - "Pillow==9.4.0", - ], - license="Copyright Scaleout Systems AB. See license for details", - zip_safe=False, - keywords="", - classifiers=[ - "Development Status :: 2 - Pre-Alpha", - "Intended Audience :: Developers", - "Natural Language :: English", - "Programming Language :: Python :: 3 :: Only", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - ], -) diff --git a/projects/static/img/logo.png b/projects/static/img/logo.png deleted file mode 100644 index c7dca1cd1755dc61280273e7e6e35ea93b8cecb9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2226 zcmZ`(X*3l47yiwd$(FZ-l!!MKvdwFWP)6C8Y-5egjJ>f;V;##FWSNvLWM3;HOOvu^ z$XeD)^&%u>H^#p9P9Ogt{vV#_InO=!IrrRi?zt3`n>t(^0vrGUE*&CN;&}i`0RR9 zH2|QuO$8jCV1tjsQUZ

s+mODn6l?u6F7?TCoazNBuo zx3~YaqE=eN_VDnq?frmTKtL1(O;meo5rI9OO(Nz5!Z z!}^~=7#umPDtqJB9pZgjNAHiRSr@+;uJcHQrd@GWeRB5m=hOyn@f*fy->F~ouKqCv zrPXjD6#<04hPiuvOE9Rm_FpINyET{Cqxu)7-^ z`xsA(#}VUx%`bBNeJ$%r)x^(V&noMtXBVW9HsV(;7nkWpFKVu%?i7^17Ev_b*xb^! z@!r|pJ$CWBr3-OxVL3WE@B9_>M>(Yr)1PWtc`VZzuN&KuW^Vb5n3~1p^z3o5 z>rqL0@;bM@L*uorJx|H#nPU7)UN*ez9!SkA>;Lp`QdWt70pu_F>w?T8tFNAs7VGp8HgNvbIq7>c0Kob6 zG&N9x!%HK34!jn;fSxo*M~qq%OX_6Y>SL!OVQG&~vbE}`6k=gt($LeKpTuyzbSd&H89q5z?y~I9 zbZ+weD^=6W3O;8jkXrP!y4vnE+*XJLgK1v9u6Hhr$!K7ih(L$b72!whpTj32i0l`n z>v95${~~i%S4>ftO}aNfgHsW~iTVd)B?~bTEa1k3tY+Lw+((vE(4|A6Kw{`IMe>D# z5M1!!ZcbnGlrjqfp1DbjJci+Fh=mA-bMG{4q0TFPUVI9t!?hKba4QDJv)_z4k8@y6 z59mQ7Tv#z_S`v-zcX*_wj*>LsQ~QWGo1pE{J>S_A-{}r1$*0+7{;mtJ7LBYgoz) z{x5CjK5opdqYlpcp3f$>ZT5Q!OOx-aySXYYiXK70hGePp4uwammMOP<=nSLC&tS%$ zxc9+&L-Jq>^!XI54^0h!y)C8=w+o*M{%JJgIyu|Xmx}_h{*IQJn&Fy8`(}dp_D%Jg z0ae=ECmY@|h$fUBJVYKro=b$9uuTCNRFB*^VXsH7Mhb{XCbH^sumJ*;gPcN!iHPPy zblGCT7#s!qq1++|-&T`(dV9r;n*vOXDM%0)7B-~mIW{N14!xGY7x{$Ia_klbZioC= zC#hRVSXPV=4-tCmme8DLLSqeSy~CNzzFnN3@_kz(i5$jF>XqdTNdqaWXKKbf#rBA zO#-~-U-27^qx}O&$CrmbEoe*#f_i!=g@BjELx8gM2|I!;9!@rFS!qABV8=NM0dH2} zMH7}2CqX%9;K;**@$vWK5TH&V>F+-0^!&j_wLg2%{Eas1mCvu6l=yA$YYL=-dZ_=R(OEzKx6-d(DoymmbI@F#@AWJk>Jb}Acv6qM;dWi z8li-elT(qutRkl%iI7u4AP8!q-2W5s^mcN0A^!gW3#PrC894oCg1NVgZvcki3<3fI gWZb=QK8_f?vy3;vHEUT-fY}7}v~FrvTt`Rz2f(fefB*mh diff --git a/projects/tasks.py b/projects/tasks.py index 44fb8eb66..35ec099b4 100644 --- a/projects/tasks.py +++ b/projects/tasks.py @@ -1,25 +1,23 @@ import collections import json -import secrets -import string from celery import shared_task from django.apps import apps from django.conf import settings from django.contrib.auth import get_user_model -import apps.tasks as apptasks -from apps.controller import delete -from apps.helpers import create_app_instance +from apps.app_registry import APP_REGISTRY +from apps.helpers import create_instance_from_form +from apps.models import BaseAppInstance, VolumeInstance +from apps.tasks import delete_resource from studio.utils import get_logger from .exceptions import ProjectCreationException -from .models import S3, Environment, Flavor, MLFlow, Project +from .models import Environment, Flavor, Project logger = get_logger(__name__) Apps = apps.get_model(app_label=settings.APPS_MODEL) -AppInstance = apps.get_model(app_label=settings.APPINSTANCE_MODEL) User = get_user_model() @@ -27,165 +25,156 @@ @shared_task def create_resources_from_template(user, project_slug, template): logger.info("Create Resources From Project Template...") + + project = Project.objects.get(slug=project_slug) + decoder = json.JSONDecoder(object_pairs_hook=collections.OrderedDict) parsed_template = template.replace("'", '"') template = decoder.decode(parsed_template) - alphabet = string.ascii_letters + string.digits - project = Project.objects.get(slug=project_slug) - logger.info("Parsing template...") - for key, item in template.items(): - logger.info("Key %s", key) - if "flavors" == key: - flavors = item - logger.info("Flavors: %s", flavors) - # TODO: potential bug. This for statement overrides variables in the outer loop. - for key, item in flavors.items(): - flavor = Flavor( - name=key, - cpu_req=item["cpu"]["requirement"], - cpu_lim=item["cpu"]["limit"], - mem_req=item["mem"]["requirement"], - mem_lim=item["mem"]["limit"], - gpu_req=item["gpu"]["requirement"], - gpu_lim=item["gpu"]["limit"], - ephmem_req=item["ephmem"]["requirement"], - ephmem_lim=item["ephmem"]["limit"], - project=project, - ) - flavor.save() - elif "environments" == key: - environments = item - logger.info("Environments: %s", environments) - for key, item in environments.items(): - try: - app = Apps.objects.filter(slug=item["app"]).order_by("-revision")[0] - except Exception as err: - logger.error( - ("App for environment not found. item.app=%s project_slug=%s " "user=%s err=%s"), - item["app"], - project_slug, - user, - err, - exc_info=True, - ) - raise - try: - environment = Environment( - name=key, - project=project, - repository=item["repository"], - image=item["image"], - app=app, - ) - environment.save() - except Exception as err: - logger.error( - ( - "Failed to create new environment: " - "key=%s " - "project=%s " - "item.repository=%s " - "image=%s " - "app=%s " - "user%s " - "err=%s" - ), - key, - project, - item["repository"], - item["image"], - app, - user, - err, - exc_info=True, - ) - elif "apps" == key: - apps = item - logger.info("Apps: %s", apps) - for key, item in apps.items(): - app_name = key - data = {"app_name": app_name, "app_action": "Create"} - if "credentials.access_key" in item: - item["credentials.access_key"] = "".join(secrets.choice(alphabet) for i in range(8)) - if "credentials.secret_key" in item: - item["credentials.secret_key"] = "".join(secrets.choice(alphabet) for i in range(14)) - if "credentials.username" in item: - item["credentials.username"] = "admin" - if "credentials.password" in item: - item["credentials.password"] = "".join(secrets.choice(alphabet) for i in range(14)) - - data = {**data, **item} - logger.info("DATA TEMPLATE") - logger.info(data) - - user_obj = User.objects.get(username=user) - - app = Apps.objects.filter(slug=item["slug"]).order_by("-revision")[0] - ( - successful, - _, - _, - ) = create_app_instance( - user=user_obj, - project=project, - app=app, - app_settings=app.settings, - data=data, - wait=True, - ) - - if not successful: - logger.error("create_app_instance failed") - raise ProjectCreationException - - elif "settings" == key: - logger.info("PARSING SETTINGS") - logger.info("Settings: %s", settings) - if "project-S3" in item: - logger.info("SETTING DEFAULT S3") - s3storage = item["project-S3"] - # Add logics: here it is referring to minio basically. - # It is assumed that minio exist, but if it doesn't - # then it blows up of course - s3obj = S3.objects.get(name=s3storage, project=project) - project.s3storage = s3obj - project.save() - if "project-MLflow" in item: - logger.info("SETTING DEFAULT MLflow") - mlflow = item["project-MLflow"] - mlflowobj = MLFlow.objects.get(name=mlflow, project=project) - project.mlflow = mlflowobj - project.save() + logger.info("Creating Project Flavors...") + flavor_dict = template.get("flavors", {}) + for flavor_name, resources in flavor_dict.items(): + logger.info("Creating flavor ") + logger.info(f"Creating flavor: {flavor_name}") + flavor = Flavor( + name=flavor_name, + cpu_req=resources["cpu"]["requirement"], + cpu_lim=resources["cpu"]["limit"], + mem_req=resources["mem"]["requirement"], + mem_lim=resources["mem"]["limit"], + gpu_req=resources["gpu"]["requirement"], + gpu_lim=resources["gpu"]["limit"], + ephmem_req=resources["ephmem"]["requirement"], + ephmem_lim=resources["ephmem"]["limit"], + project=project, + ) + flavor.save() + + logger.info("Creating Project Volumes...") + volume_dict = template.get("volumes", {}) + for volume_name, data in volume_dict.items(): + data["name"] = volume_name + logger.info(f"Creating volume using {data}") + form_class = APP_REGISTRY.get_form_class("volumeK8s") + form = form_class(data, project_pk=project.pk) + if form.is_valid(): + create_instance_from_form(form, project, "volumeK8s") + else: + logger.error(f"Form is invalid: {form.errors.as_data()}") + raise ProjectCreationException(f"Form is invalid: {form.errors.as_data()}") + + apps_dict = template.get("apps", {}) + logger.info("Initiate Creation of Project Apps...") + form_dict = {} + for app_slug, data in apps_dict.items(): + logger.info(f"Creating {app_slug} using raw data {data}") + + # Handle mounting of volumes + if "volume" in data: + try: + volumes = VolumeInstance.objects.filter(project=project, name=data["volume"]) + data["volume"] = volumes + logger.info(f"Modified data with specified volume: {data}") + except VolumeInstance.DoesNotExist: + raise ProjectCreationException(f"Volume {data['volume']} not found") + + # Handle flavor selection + if "flavor" in data: + try: + # Get the only flavor that matches the project and the name + flavor = Flavor.objects.filter(project__pk=project.pk, name=data["flavor"]).first() + data["flavor"] = flavor + logger.info(f"Modified data with specified flavor: {data}") + except Flavor.DoesNotExist: + raise ProjectCreationException(f"Flavor {data['flavor']} not found") + + model_form_tuple = APP_REGISTRY.get(app_slug) + + # Check if the model form tuple exists + if not model_form_tuple: + logger.error(f"App {app_slug} not found") + raise ProjectCreationException(f"App {app_slug} not found") + + # Create form + form = model_form_tuple.Form(data, project_pk=project.pk) + + # Handle validation + if form.is_valid(): + logger.info("Form is valid - Appending form to list") + form_dict[app_slug] = form else: - logger.error("Template has either not valid or unknown keys") - raise ProjectCreationException + logger.error(f"Form is invalid: {form.errors.as_data()}") + raise ProjectCreationException(f"Form is invalid: {form.errors.as_data()}") + + # All forms are valid, lets create apps. + logger.info("All forms valid, creating apps...") + for app_slug, form in form_dict.items(): + create_instance_from_form(form, project, app_slug) + + env_dict = template.get("environments", {}) + logger.info("Creating Project Environments...") + for name, env_settings in env_dict.items(): + try: + app = Apps.objects.filter(slug=env_settings["app"]).order_by("-revision")[0] + except Exception as err: + logger.error( + ("App for environment not found. item.app=%s project_slug=%s " "user=%s err=%s"), + env_settings["app"], + project_slug, + user, + err, + exc_info=True, + ) + raise + try: + environment = Environment( + name=name, + project=project, + repository=env_settings["repository"], + image=env_settings["image"], + app=app, + ) + environment.save() + except Exception as err: + logger.error( + ( + "Failed to create new environment: " + "key=%s " + "project=%s " + "item.repository=%s " + "image=%s " + "app=%s " + "user%s " + "err=%s" + ), + name, + project, + settings["repository"], + settings["image"], + app, + user, + err, + exc_info=True, + ) project.status = "active" project.save() -@shared_task -def delete_project_apps(project_slug): - project = Project.objects.get(slug=project_slug) - apps = AppInstance.objects.filter(project=project) - for app in apps: - apptasks.delete_resource.delay(app.pk) - - @shared_task def delete_project(project_pk): logger.info("SCHEDULING DELETION OF ALL INSTALLED APPS") project = Project.objects.get(pk=project_pk) - delete_project_apps_permanently(project) + delete_project_apps(project) project.delete() @shared_task -def delete_project_apps_permanently(project): - apps = AppInstance.objects.filter(project=project) - - for app in apps: - helm_output = delete(app.parameters) - logger.info(helm_output.stderr.decode("utf-8")) +def delete_project_apps(project): + for orm_model in APP_REGISTRY.iter_orm_models(): + queryset = orm_model.objects.filter(project=project) + for instance in queryset: + serialized_instance = instance.serialize() + delete_resource(serialized_instance) diff --git a/projects/tests/test_create_mlflow.py b/projects/tests/test_create_mlflow.py deleted file mode 100644 index e8707ee76..000000000 --- a/projects/tests/test_create_mlflow.py +++ /dev/null @@ -1,52 +0,0 @@ -from django.contrib.auth import get_user_model -from django.test import TestCase - -from ..models import MLFlow, Project - -User = get_user_model() - - -class CreateMLFlowTestCase(TestCase): - project_name = "test-perm-mlflow" - - def setUp(self) -> None: - self.user = User.objects.create_user("foo1", "foo@test.com", "bar") - self.project = Project.objects.create_project(name=self.project_name, owner=self.user, description="") - - def test_no_default_for_project(self): - obj = MLFlow( - name="mlflow1", - project=self.project, - owner=self.user, - ) - - obj.save() - - project = Project.objects.get(name=self.project_name) - - self.assertEqual(project.mlflow.name, obj.name) - - def test_default_existis_for_project(self): - obj = MLFlow( - name="mlflow1", - project=self.project, - owner=self.user, - ) - - obj.save() - - project = Project.objects.get(name=self.project_name) - - self.assertEqual(project.mlflow.name, obj.name) - - obj2 = MLFlow( - name="mlflow2", - project=self.project, - owner=self.user, - ) - - obj2.save() - - project = Project.objects.get(name=self.project_name) - - self.assertEqual(project.mlflow.name, obj.name) diff --git a/projects/tests/test_project.py b/projects/tests/test_project.py index 640d6b86d..59bfa5a0c 100644 --- a/projects/tests/test_project.py +++ b/projects/tests/test_project.py @@ -1,9 +1,10 @@ +import base64 + from django.contrib.auth import get_user_model from django.contrib.auth.models import Permission from django.contrib.contenttypes.models import ContentType from django.test import TestCase, override_settings -from ..helpers import decrypt_key from ..models import Project User = get_user_model() @@ -29,6 +30,11 @@ def setUp(self): def test_decrypt_key(self): project = Project.objects.filter(name="test-secret").first() + def decrypt_key(key): + base64_bytes = key.encode("ascii") + result = base64.b64decode(base64_bytes) + return result.decode("ascii") + self.assertEqual(decrypt_key(project.project_key), "key") self.assertEqual(decrypt_key(project.project_secret), "secret") diff --git a/projects/tests/test_project_view.py b/projects/tests/test_project_view.py index e1ddef23b..41992731b 100644 --- a/projects/tests/test_project_view.py +++ b/projects/tests/test_project_view.py @@ -82,33 +82,3 @@ def test_forbidden_project_delete(self): ) self.assertTemplateUsed(response, "403.html") self.assertEqual(response.status_code, 403) - - def test_forbidden_project_setS3storage(self): - """ - Test non-project member not allowed to access project setS3storage - """ - self.client.login(username=test_member["email"], password=test_member["password"]) - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "projects:set_s3storage", - kwargs={"project_slug": project.slug}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) - - def test_forbidden_project_setmlflow(self): - """ - Test non-project member not allowed to access project setmlflow - """ - self.client.login(username=test_member["email"], password=test_member["password"]) - project = Project.objects.get(name="test-perm") - response = self.client.get( - reverse( - "projects:set_mlflow", - kwargs={"project_slug": project.slug}, - ) - ) - self.assertTemplateUsed(response, "403.html") - self.assertEqual(response.status_code, 403) diff --git a/projects/urls.py b/projects/urls.py index 21496c44f..aabd02984 100644 --- a/projects/urls.py +++ b/projects/urls.py @@ -33,12 +33,6 @@ ), path("/settings/", views.settings, name="settings"), path("/delete/", views.delete, name="delete"), - path( - "/setS3storage/", - views.set_s3storage, - name="set_s3storage", - ), - path("/setmlflow/", views.set_mlflow, name="set_mlflow"), path( "/details/change/", views.change_description, diff --git a/projects/views.py b/projects/views.py index 52379ae50..a408b16ed 100644 --- a/projects/views.py +++ b/projects/views.py @@ -22,22 +22,16 @@ from guardian.decorators import permission_required_or_403 from guardian.shortcuts import assign_perm, remove_perm +from apps.app_registry import APP_REGISTRY +from apps.models import BaseAppInstance + from .exceptions import ProjectCreationException from .forms import PublishProjectToGitHub -from .models import ( - S3, - Environment, - Flavor, - MLFlow, - Project, - ProjectLog, - ProjectTemplate, -) +from .models import Environment, Flavor, Project, ProjectLog, ProjectTemplate from .tasks import create_resources_from_template, delete_project logger = logging.getLogger(__name__) Apps = apps.get_model(app_label=django_settings.APPS_MODEL) -AppInstance = apps.get_model(app_label=django_settings.APPINSTANCE_MODEL) AppCategories = apps.get_model(app_label=django_settings.APPCATEGORIES_MODEL) Model = apps.get_model(app_label=django_settings.MODELS_MODEL) User = get_user_model() @@ -51,7 +45,7 @@ def get(self, request): projects = Project.objects.filter(status="active").distinct("pk") else: projects = Project.objects.filter( - Q(owner=request.user) | Q(authorized=request.user), + Q(owner=request.user.id) | Q(authorized=request.user.id), status="active", ).distinct("pk") except TypeError as err: @@ -110,9 +104,7 @@ def settings(request, project_slug): environments = Environment.objects.filter(project=project) apps = Apps.objects.all().order_by("slug", "-revision").distinct("slug") - s3instances = S3.objects.filter(Q(project=project), Q(app__state="Running")) flavors = Flavor.objects.filter(project=project) - mlflows = MLFlow.objects.filter(Q(project=project), Q(app__state="Running")) return render(request, template, locals()) @@ -275,71 +267,6 @@ def delete_flavor(request, project_slug): ) -@login_required -@permission_required_or_403("can_view_project", (Project, "slug", "project_slug")) -def set_s3storage(request, project_slug, s3storage=[]): - # TODO: Ensure that the user has the correct permissions to set - # this specific - # s3 object to storage in this project (need to check that - # the user has access to the - # project as well.) - if request.method == "POST" or s3storage: - project = Project.objects.get(slug=project_slug) - - if s3storage: - s3obj = S3.objects.get(name=s3storage, project=project) - else: - pk = request.POST.get("s3storage") - if pk == "blank": - s3obj = None - else: - s3obj = S3.objects.get(pk=pk) - - project.s3storage = s3obj - project.save() - - if s3storage: - return JsonResponse({"status": "ok"}) - - return HttpResponseRedirect( - reverse( - "projects:settings", - kwargs={"project_slug": project.slug}, - ) - ) - - -@login_required -@permission_required_or_403("can_view_project", (Project, "slug", "project_slug")) -def set_mlflow(request, project_slug, mlflow=[]): - # TODO: Ensure that the user has the correct permissions - # to set this specific - # MLFlow object to MLFlow Server in this project (need to check - # that the user has access to the - # project as well.) - if request.method == "POST" or mlflow: - project = Project.objects.get(slug=project_slug) - - if mlflow: - mlflowobj = MLFlow.objects.get(name=mlflow, project=project) - else: - pk = request.POST.get("mlflow") - mlflowobj = MLFlow.objects.get(pk=pk) - - project.mlflow = mlflowobj - project.save() - - if mlflow: - return JsonResponse({"status": "ok"}) - - return HttpResponseRedirect( - reverse( - "projects:settings", - kwargs={"project_slug": project.slug}, - ) - ) - - @method_decorator( permission_required_or_403("can_view_project", (Project, "slug", "project_slug")), name="dispatch", @@ -532,72 +459,51 @@ class DetailsView(View): template_name = "projects/overview.html" def get(self, request, project_slug): - resources = list() - models = Model.objects.none() + project = Project.objects.get(slug=project_slug) + resources = [] app_ids = [] - project = None - filemanager_instance = None + if request.user.is_superuser: + categories = AppCategories.objects.all().order_by("-priority") + else: + categories = AppCategories.objects.all().exclude(slug__in=["admin-apps"]).order_by("-priority") - if request.user.is_authenticated: - project = Project.objects.get(slug=project_slug) - if request.user.is_superuser: - categories = AppCategories.objects.all().order_by("-priority") - else: - categories = AppCategories.objects.all().exclude(slug__in=["admin-apps"]).order_by("-priority") - # models = Model.objects.filter(project=project).order_by("-uploaded_at")[:10] - models = Model.objects.filter(project=project).order_by("-uploaded_at") - - def filter_func(slug): - return Q(app__category__slug=slug) - - for category in categories: - app_instances_of_category = AppInstance.objects.get_app_instances_of_project( - user=request.user, - project=project, - filter_func=filter_func(slug=category.slug), - # limit=5, - ) + for category in categories: + # Get all subclasses of Base - app_ids += [obj.id for obj in app_instances_of_category] + instances_per_category_list = [] + for orm_model in APP_REGISTRY.iter_orm_models(): + # Filter instances of each subclass by project, user and status. + # See the get_app_instances_of_project_filter method in base.py - apps_of_category = ( - Apps.objects.filter(category=category, user_can_create=True) - .order_by("slug", "-revision") - .distinct("slug") + queryset_per_category = orm_model.objects.get_app_instances_of_project( + user=request.user, project=project, filter_func=Q(app__category__slug=category.slug) ) - resources.append( - { - "title": category.name, - "objs": app_instances_of_category, - "apps": apps_of_category, - } - ) + if queryset_per_category: + app_ids += [obj.id for obj in queryset_per_category] + instances_per_category_list.extend([instance for instance in queryset_per_category]) - def filter_app_slug(slug): - return Q(app__slug=slug) + # Filter the available apps specified in the project template + available_apps = [app.pk for app in project.project_template.available_apps.all()] - filemanager_instance = AppInstance.objects.get_app_instances_of_project( - user=request.user, project=project, filter_func=filter_app_slug(slug="filemanager") - ).first() + apps_per_category = ( + Apps.objects.filter(category=category, user_can_create=True, pk__in=available_apps) + .order_by("slug", "-revision") + .distinct("slug") + ) - if filemanager_instance: - creation_date = filemanager_instance.created_on - now = datetime.datetime.now(datetime.timezone.utc) - age = now - creation_date - timedelta = datetime.timedelta(hours=24) - hours = timedelta - age - hours = round(hours.total_seconds() / 3600) - else: - hours = 0 + resources.append( + { + "title": category.name, + "instances": instances_per_category_list, + "apps": apps_per_category, + } + ) context = { "resources": resources, - "models": models, "project": project, "app_ids": app_ids, - "filemanager_instance": filemanager_instance, - "hours": hours, } return render( diff --git a/pyproject.toml b/pyproject.toml index 8609df636..df8340866 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -28,6 +28,7 @@ django-tagulous = "==1.3.3" django-guardian = "==2.4.0" djangorestframework = "== 3.15.1" django-crispy-forms = "==2.1" +crispy-bootstrap5 = "==2024.2" # django-wiki wiki = "==0.11.1" @@ -109,7 +110,7 @@ python_version = "3.12" disallow_subclassing_any = false ignore_missing_imports = false warn_return_any = true -untyped_calls_exclude = ["apps.helpers","projects.models"] +untyped_calls_exclude = ["projects.models"] # ["apps.helpers","projects.models"] exclude = ["venv", ".venv", "examples"] #plugins = ["mypy_django_plugin.main"] @@ -137,7 +138,7 @@ module = [ "*.tests.*", "*.tests", "conftest", - "cypress.*" + #"cypress.*" ] check_untyped_defs = false disallow_incomplete_defs = false @@ -150,6 +151,16 @@ disallow_untyped_defs = false module = "*.migrations.*" ignore_errors = true +# A temporary section to ease the transition of mypy into the codebase +[[tool.mypy.overrides]] +module = [ + "apps.*", + "cypress.*", + "cypress/e2e/setup-scripts/seed_collections_user" +] +strict = false +ignore_errors = true + # A temporary section to ease the transition of mypy into the codebase [[tool.mypy.overrides]] module = [ @@ -178,6 +189,7 @@ no_implicit_reexport = false module = [ "boto3.*", "colorlog.*", + "crispy_forms.*", "django.*", "django_structlog.*", "rest_framework.*", diff --git a/scripts/app_instance_permissions.py b/scripts/app_instance_permissions.py index 9f0b64510..bf8a4bbca 100644 --- a/scripts/app_instance_permissions.py +++ b/scripts/app_instance_permissions.py @@ -1,20 +1,21 @@ from guardian.shortcuts import assign_perm, remove_perm -from apps.models import AppInstance +from apps.app_registry import APP_REGISTRY +from apps.models import BaseAppInstance def run(*args): """Reads all AppInstance objects and sets correct permission based on owner (user) and the instance access property""" - app_instances_all = AppInstance.objects.all() + for orm_model in APP_REGISTRY.iter_orm_models(): + app_instances_all = orm_model.objects.all() + for app_instance in app_instances_all: + owner = app_instance.owner - for app_instance in app_instances_all: - owner = app_instance.owner - - if app_instance.access == "private": - if not owner.has_perm("can_access_app", app_instance): - assign_perm("can_access_app", owner, app_instance) - else: - if owner.has_perm("can_access_app", app_instance): - remove_perm("can_access_app", owner, app_instance) + if getattr(app_instance, "access", False) == "private": + if not owner.has_perm("can_access_app", app_instance): + assign_perm("can_access_app", owner, app_instance) + else: + if owner.has_perm("can_access_app", app_instance): + remove_perm("can_access_app", owner, app_instance) diff --git a/static/css/serve-elements.css b/static/css/serve-elements.css index 10ab1ead0..da4aad97f 100644 --- a/static/css/serve-elements.css +++ b/static/css/serve-elements.css @@ -84,6 +84,27 @@ border-color: var(--scaleout-green); } +.form-control-with-spinner { + width: -webkit-calc(100% - 50px); + width: -moz-calc(100% - 50px); + width: calc(100% - 50px); +} + +.client-validation-feedback { + display: block; + width: 100%; + margin-top: 0.25rem; + font-size: 80%; +} + +.client-validation-valid { + color: rgb(25, 135, 84); +} + +.client-validation-invalid { + color: var(--bs-danger); +} + /* links */ a { diff --git a/studio/migrations/__init__.py b/studio/migrations/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/migrations/apps/0001_initial.py b/studio/migrations/apps/0001_initial.py deleted file mode 100644 index 3d8fee336..000000000 --- a/studio/migrations/apps/0001_initial.py +++ /dev/null @@ -1,128 +0,0 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 - -import django.db.models.deletion -import tagulous.models.models -from django.db import migrations, models - - -class Migration(migrations.Migration): - initial = True - - dependencies = [] - - operations = [ - migrations.CreateModel( - name="AppCategories", - fields=[ - ("name", models.CharField(max_length=512)), - ("priority", models.IntegerField(default=100)), - ("slug", models.CharField(default="", max_length=512, primary_key=True, serialize=False)), - ], - ), - migrations.CreateModel( - name="AppInstance", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("access", models.CharField(blank=True, default="private", max_length=20, null=True)), - ("created_on", models.DateTimeField(auto_now_add=True)), - ("deleted_on", models.DateTimeField(blank=True, null=True)), - ("description", models.TextField(blank=True, default="", null=True)), - ("info", models.JSONField(blank=True, null=True)), - ("name", models.CharField(default="app_name", max_length=512)), - ("parameters", models.JSONField(blank=True, null=True)), - ("state", models.CharField(blank=True, max_length=50, null=True)), - ("table_field", models.JSONField(blank=True, null=True)), - ("updated_on", models.DateTimeField(auto_now=True)), - ], - options={ - "permissions": [("can_access_app", "Can access app service")], - }, - ), - migrations.CreateModel( - name="Tagulous_AppInstance_tags", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("name", models.CharField(max_length=255, unique=True)), - ("slug", models.SlugField()), - ( - "count", - models.IntegerField(default=0, help_text="Internal counter of how many times this tag is in use"), - ), - ( - "protected", - models.BooleanField(default=False, help_text="Will not be deleted when the count reaches 0"), - ), - ], - options={ - "ordering": ("name",), - "abstract": False, - "unique_together": {("slug",)}, - }, - bases=(tagulous.models.models.BaseTagModel, models.Model), - ), - migrations.CreateModel( - name="ResourceData", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("cpu", models.IntegerField()), - ("gpu", models.IntegerField()), - ("mem", models.IntegerField()), - ("time", models.IntegerField()), - ( - "appinstance", - models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="resourcedata", to="apps.appinstance" - ), - ), - ], - ), - migrations.CreateModel( - name="AppStatus", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("info", models.JSONField(blank=True, null=True)), - ("status_type", models.CharField(default="app_name", max_length=15)), - ("time", models.DateTimeField(auto_now_add=True)), - ( - "appinstance", - models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="status", to="apps.appinstance" - ), - ), - ], - options={ - "get_latest_by": "time", - }, - ), - migrations.CreateModel( - name="Apps", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("user_can_create", models.BooleanField(default=True)), - ("user_can_edit", models.BooleanField(default=True)), - ("user_can_delete", models.BooleanField(default=True)), - ("access", models.CharField(blank=True, default="public", max_length=20, null=True)), - ("chart", models.CharField(max_length=512)), - ("chart_archive", models.FileField(blank=True, null=True, upload_to="apps/")), - ("created_on", models.DateTimeField(auto_now_add=True)), - ("description", models.TextField(blank=True, default="", null=True)), - ("logo", models.CharField(blank=True, max_length=512, null=True)), - ("name", models.CharField(max_length=512)), - ("priority", models.IntegerField(default=100)), - ("revision", models.IntegerField(default=1)), - ("settings", models.JSONField(blank=True, null=True)), - ("slug", models.CharField(blank=True, max_length=512, null=True)), - ("table_field", models.JSONField(blank=True, null=True)), - ("updated_on", models.DateTimeField(auto_now=True)), - ( - "category", - models.ForeignKey( - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="apps", - to="apps.appcategories", - ), - ), - ], - ), - ] diff --git a/studio/migrations/apps/0002_initial.py b/studio/migrations/apps/0002_initial.py deleted file mode 100644 index ea26d718d..000000000 --- a/studio/migrations/apps/0002_initial.py +++ /dev/null @@ -1,83 +0,0 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 - -import django.db.models.deletion -import tagulous.models.fields -from django.conf import settings -from django.db import migrations, models - - -class Migration(migrations.Migration): - initial = True - - dependencies = [ - ("models", "0001_initial"), - ("apps", "0001_initial"), - ("projects", "0001_initial"), - migrations.swappable_dependency(settings.AUTH_USER_MODEL), - ] - - operations = [ - migrations.AddField( - model_name="apps", - name="projects", - field=models.ManyToManyField(blank=True, to="projects.project"), - ), - migrations.AddField( - model_name="appinstance", - name="app", - field=models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="appinstance", to="apps.apps" - ), - ), - migrations.AddField( - model_name="appinstance", - name="app_dependencies", - field=models.ManyToManyField(blank=True, to="apps.appinstance"), - ), - migrations.AddField( - model_name="appinstance", - name="flavor", - field=models.ForeignKey( - null=True, - on_delete=django.db.models.deletion.RESTRICT, - related_name="appinstance", - to="projects.flavor", - ), - ), - migrations.AddField( - model_name="appinstance", - name="model_dependencies", - field=models.ManyToManyField(blank=True, to="models.model"), - ), - migrations.AddField( - model_name="appinstance", - name="owner", - field=models.ForeignKey( - null=True, - on_delete=django.db.models.deletion.CASCADE, - related_name="app_owner", - to=settings.AUTH_USER_MODEL, - ), - ), - migrations.AddField( - model_name="appinstance", - name="project", - field=models.ForeignKey( - on_delete=django.db.models.deletion.CASCADE, related_name="appinstance", to="projects.project" - ), - ), - migrations.AddField( - model_name="appinstance", - name="tags", - field=tagulous.models.fields.TagField( - _set_tag_meta=True, - blank=True, - help_text="Enter a comma-separated tag string", - to="apps.tagulous_appinstance_tags", - ), - ), - migrations.AlterUniqueTogether( - name="apps", - unique_together={("slug", "revision")}, - ), - ] diff --git a/studio/migrations/apps/0003_appinstance_note_on_linkonly_privacy.py b/studio/migrations/apps/0003_appinstance_note_on_linkonly_privacy.py deleted file mode 100644 index f59f709b6..000000000 --- a/studio/migrations/apps/0003_appinstance_note_on_linkonly_privacy.py +++ /dev/null @@ -1,17 +0,0 @@ -# Generated by Django 4.2.7 on 2024-02-13 16:49 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("apps", "0002_initial"), - ] - - operations = [ - migrations.AddField( - model_name="appinstance", - name="note_on_linkonly_privacy", - field=models.TextField(blank=True, default="", null=True), - ), - ] diff --git a/studio/migrations/apps/0004_remove_appinstance_note_on_linkonly_privacy.py b/studio/migrations/apps/0004_remove_appinstance_note_on_linkonly_privacy.py deleted file mode 100644 index 5cfd0c7bc..000000000 --- a/studio/migrations/apps/0004_remove_appinstance_note_on_linkonly_privacy.py +++ /dev/null @@ -1,16 +0,0 @@ -# Generated by Django 4.2.7 on 2024-02-15 17:06 - -from django.db import migrations - - -class Migration(migrations.Migration): - dependencies = [ - ("apps", "0003_appinstance_note_on_linkonly_privacy"), - ] - - operations = [ - migrations.RemoveField( - model_name="appinstance", - name="note_on_linkonly_privacy", - ), - ] diff --git a/studio/migrations/apps/0005_appinstance_note_on_linkonly_privacy.py b/studio/migrations/apps/0005_appinstance_note_on_linkonly_privacy.py deleted file mode 100644 index 2486663bc..000000000 --- a/studio/migrations/apps/0005_appinstance_note_on_linkonly_privacy.py +++ /dev/null @@ -1,17 +0,0 @@ -# Generated by Django 4.2.7 on 2024-02-21 12:15 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("apps", "0004_remove_appinstance_note_on_linkonly_privacy"), - ] - - operations = [ - migrations.AddField( - model_name="appinstance", - name="note_on_linkonly_privacy", - field=models.TextField(blank=True, default="", null=True), - ), - ] diff --git a/studio/migrations/apps/0006_appinstance_collections.py b/studio/migrations/apps/0006_appinstance_collections.py deleted file mode 100644 index 78d21bbc1..000000000 --- a/studio/migrations/apps/0006_appinstance_collections.py +++ /dev/null @@ -1,18 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-20 15:02 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("collections_module", "0001_initial"), - ("apps", "0005_appinstance_note_on_linkonly_privacy"), - ] - - operations = [ - migrations.AddField( - model_name="appinstance", - name="collections", - field=models.ManyToManyField(blank=True, related_name="app_instances", to="collections_module.collection"), - ), - ] diff --git a/studio/migrations/apps/0007_appinstance_source_code_url.py b/studio/migrations/apps/0007_appinstance_source_code_url.py deleted file mode 100644 index 31a42c5b4..000000000 --- a/studio/migrations/apps/0007_appinstance_source_code_url.py +++ /dev/null @@ -1,17 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-28 15:25 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("apps", "0006_appinstance_collections"), - ] - - operations = [ - migrations.AddField( - model_name="appinstance", - name="source_code_url", - field=models.URLField(blank=True, null=True), - ), - ] diff --git a/studio/migrations/apps/__init__.py b/studio/migrations/apps/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/migrations/models/__init__.py b/studio/migrations/models/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/migrations/monitor/__init__.py b/studio/migrations/monitor/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/migrations/portal/0001_initial.py b/studio/migrations/portal/0001_initial.py deleted file mode 100644 index c5a7bf597..000000000 --- a/studio/migrations/portal/0001_initial.py +++ /dev/null @@ -1,38 +0,0 @@ -# Generated by Django 4.2.5 on 2023-11-16 08:07 - -import django.db.models.deletion -from django.db import migrations, models - - -class Migration(migrations.Migration): - initial = True - - dependencies = [ - ("models", "0001_initial"), - ("projects", "0001_initial"), - ] - - operations = [ - migrations.CreateModel( - name="PublicModelObject", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("obj", models.FileField(upload_to="models/objects/")), - ("updated_on", models.DateTimeField(auto_now=True)), - ("created_on", models.DateTimeField(auto_now_add=True)), - ("model", models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to="models.model")), - ], - ), - migrations.CreateModel( - name="PublishedModel", - fields=[ - ("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), - ("name", models.CharField(max_length=512)), - ("pattern", models.CharField(default="", max_length=255)), - ("updated_on", models.DateTimeField(auto_now=True)), - ("created_on", models.DateTimeField(auto_now_add=True)), - ("model_obj", models.ManyToManyField(to="portal.publicmodelobject")), - ("project", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="projects.project")), - ], - ), - ] diff --git a/studio/migrations/portal/0002_publishedmodel_collections.py b/studio/migrations/portal/0002_publishedmodel_collections.py deleted file mode 100644 index 790d8dc28..000000000 --- a/studio/migrations/portal/0002_publishedmodel_collections.py +++ /dev/null @@ -1,20 +0,0 @@ -# Generated by Django 4.2.7 on 2024-03-22 07:04 - -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("collections_module", "0001_initial"), - ("portal", "0001_initial"), - ] - - operations = [ - migrations.AddField( - model_name="publishedmodel", - name="collections", - field=models.ManyToManyField( - blank=True, related_name="published_models", to="collections_module.collection" - ), - ), - ] diff --git a/studio/migrations/portal/__init__.py b/studio/migrations/portal/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/migrations/projects/0002_project_project_template.py b/studio/migrations/projects/0002_project_project_template.py deleted file mode 100644 index 5661af68b..000000000 --- a/studio/migrations/projects/0002_project_project_template.py +++ /dev/null @@ -1,20 +0,0 @@ -# Generated by Django 4.2.7 on 2024-02-23 13:16 - -import django.db.models.deletion -from django.db import migrations, models - - -class Migration(migrations.Migration): - dependencies = [ - ("projects", "0001_initial"), - ] - - operations = [ - migrations.AddField( - model_name="project", - name="project_template", - field=models.ForeignKey( - null=True, on_delete=django.db.models.deletion.DO_NOTHING, to="projects.projecttemplate" - ), - ), - ] diff --git a/studio/migrations/projects/__init__.py b/studio/migrations/projects/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/studio/settings.py b/studio/settings.py index 9facb24b7..afecb5700 100644 --- a/studio/settings.py +++ b/studio/settings.py @@ -91,18 +91,15 @@ "tagulous", "guardian", "crispy_forms", + "crispy_bootstrap5", "common", "portal", "projects", "models", - "monitor", "apps", "api", - "customtags", - "news", "axes", # django-axes for brute force login protection "django_password_validators", # django-password-validators for password validation - "collections_module", ] + DJANGO_WIKI_APPS MIDDLEWARE = ( @@ -123,7 +120,9 @@ ) ROOT_URLCONF = "studio.urls" -CRISPY_TEMPLATE_PACK = "bootstrap" + +CRISPY_ALLOWED_TEMPLATE_PACKS = "bootstrap5" +CRISPY_TEMPLATE_PACK = "bootstrap5" TEMPLATES = [ { @@ -351,7 +350,7 @@ KUBECONFIG = "/app/cluster.conf" NAMESPACE = "default" KUBE_API_REQUEST_TIMEOUT = 1 -STORAGECLASS = "microk8s-hostpath" +STORAGECLASS = "local-path" # This can be simply "localhost", but it's better to test with a # wildcard dns such as nip.io @@ -378,7 +377,7 @@ # Apps APPS_MODEL = "apps.Apps" -APPINSTANCE_MODEL = "apps.AppInstance" +APPINSTANCE_MODEL = "apps.BaseAppInstance" APPCATEGORIES_MODEL = "apps.AppCategories" # Models @@ -405,18 +404,13 @@ EMAIL_PASSWORD = os.getenv("EMAIL_PASSWORD") EMAIL_HOST_PASSWORD = os.getenv("EMAIL_PASSWORD") -# 2024-02-21: Removed because this is not used. -# VERSION = "dev" MIGRATION_MODULES = { - "apps": "studio.migrations.apps", - "models": "studio.migrations.models", - "monitor": "studio.migrations.monitor", - "portal": "studio.migrations.portal", - "projects": "studio.migrations.projects", + "apps": "apps.migrations", + "models": "models.migrations", + "portal": "portal.migrations", + "projects": "projects.migrations", "common": "common.migrations", - "news": "news.migrations", - "collections_module": "collections_module.migrations", } # Defines how many apps a user is allowed to create within one project @@ -440,6 +434,7 @@ "combiner": 0, "mongodb": 0, "netpolicy": 0, + "filemanager": 1, } PROJECTS_PER_USER_LIMIT = 5 @@ -519,3 +514,5 @@ logger_factory=structlog.stdlib.LoggerFactory(), cache_logger_on_first_use=True, ) + +LOKI_SVC = None diff --git a/studio/tests.py b/studio/tests.py index 254fdf540..8974f10f1 100644 --- a/studio/tests.py +++ b/studio/tests.py @@ -5,7 +5,7 @@ from django.test import TestCase from guardian.shortcuts import assign_perm, remove_perm -from apps.models import AppInstance, Apps +from apps.models import Apps, AppStatus, JupyterInstance, Subdomain from common.models import EmailVerificationTable, UserProfile from projects.models import Project from scripts.app_instance_permissions import run @@ -21,12 +21,16 @@ def get_data(self, user, access): project = Project.objects.create_project(name="test-perm", owner=user, description="") app = Apps.objects.create(name="FEDn Combiner") - app_instance = AppInstance.objects.create( + subdomain = Subdomain.objects.create(subdomain="test_internal") + app_status = AppStatus.objects.create(status="Created") + app_instance = JupyterInstance.objects.create( access=access, owner=user, name="test_app_instance_private", app=app, project=project, + subdomain=subdomain, + app_status=app_status, ) return [project, app, app_instance] diff --git a/studio/urls.py b/studio/urls.py index 77e99a014..dfe4e5303 100644 --- a/studio/urls.py +++ b/studio/urls.py @@ -47,12 +47,6 @@ path("", include("common.urls", namespace="common")), path("", include("models.urls", namespace="models")), path("", include("portal.urls", namespace="portal")), - path("", include("news.urls", namespace="news")), - path("", include("collections_module.urls", namespace="collections_module")), - path( - "//monitor/", - include("monitor.urls", namespace="monitor"), - ), path("projects//apps/", include("apps.urls", namespace="apps")), ] + staticfiles_urlpatterns() diff --git a/studio/views.py b/studio/views.py index ed75eeeae..e433b1b35 100644 --- a/studio/views.py +++ b/studio/views.py @@ -1,5 +1,4 @@ -import json -from typing import Any, cast +from typing import Any, Callable, cast import requests from django.conf import settings @@ -9,20 +8,17 @@ from django.core.exceptions import ObjectDoesNotExist from django.core.mail import send_mail from django.db.models import Q -from django.db.models.signals import post_save, pre_save +from django.db.models.signals import pre_save from django.dispatch import receiver from django.http import HttpResponse, HttpResponseRedirect from django.shortcuts import render, reverse -from rest_framework.authentication import ( - BasicAuthentication, - SessionAuthentication, - TokenAuthentication, -) +from rest_framework.authentication import SessionAuthentication, TokenAuthentication from rest_framework.permissions import BasePermission, IsAuthenticated from rest_framework.response import Response from rest_framework.views import APIView -from apps.models import AppInstance +from apps.app_registry import APP_REGISTRY +from apps.models import BaseAppInstance, Subdomain from common.models import UserProfile from models.models import Model from projects.models import Project @@ -33,7 +29,21 @@ logger = get_logger(__name__) +def disable_for_loaddata(signal_handler: Callable[..., Any]) -> Callable[..., Any]: + """ + Decorator that turns off signal handlers when loading fixture data. + """ + + def wrapper(*args: Any, **kwargs: Any) -> Any: + if kwargs.get("raw", False): + return + return signal_handler(*args, **kwargs) + + return wrapper + + @receiver(pre_save, sender=User) +@disable_for_loaddata def set_new_user_inactive(sender: Model, instance: User, **kwargs: dict[str, Any]) -> None: if instance._state.adding and settings.INACTIVE_USERS and not instance.is_superuser: logger.info("Creating Inactive User") @@ -54,19 +64,29 @@ def has_permission(self, request: Response, view: object) -> bool: """ Should simply return, or raise a 403 response. """ + release = request.GET.get("release", None) try: - release = request.GET.get("release") - app_instance = AppInstance.objects.filter(parameters__contains={"release": release}).last() - project = app_instance.project + # Must fetch the subdomain and reverse to the related model. + subdomain = Subdomain.objects.get(subdomain=release) + instance = BaseAppInstance.objects.filter(subdomain=subdomain).last() + project = instance.project # TODO: Make it an explicit exception. At least catch `Exception` except: # noqa: E722 project_slug = request.GET.get("project") project = Project.objects.get(slug=project_slug) return cast(bool, request.user.has_perm("can_view_project", project)) - if app_instance.access == "private": - return cast(bool, app_instance.owner == request.user) - elif app_instance.access == "project": + model_class = APP_REGISTRY.get_orm_model(instance.app.slug) + if model_class is None: + return False + instance = getattr(instance, model_class.__name__.lower()) + access = getattr(instance, "access", None) + + if access is None: + return False + elif instance.access == "private": + return cast(bool, instance.owner == request.user) + elif instance.access == "project": return cast(bool, request.user.has_perm("can_view_project", project)) else: return True diff --git a/templates/apps/create.html b/templates/apps/create.html deleted file mode 100644 index a32c9a32f..000000000 --- a/templates/apps/create.html +++ /dev/null @@ -1,511 +0,0 @@ -{% extends 'base.html' %} -{% block title %}Create {{ app.name }}{% endblock %} -{% load static %} -{% load custom_tags %} - -{% block content %} -

- -

Create {{ app.name }}

-
-
- - {% if app.slug == 'customapp' %} -

This form allows you to start hosting a custom app that fulfills certain requirements on SciLifeLab Serve (the app itself can be built on any framework). Please read our documentation page on custom apps for the list of requirements and step-by-step instructions on deployment.

- {% elif app.slug == 'dashapp' %} -

This form allows you to start hosting a Dash app at SciLifeLab Serve. Please read our documentation page on Dash apps for step-by-step instructions.

- {% elif app.slug in 'shinyapp,shinyproxyapp' %} -

This form allows you to start hosting a Shiny app at SciLifeLab Serve. Please read our documentation page on Shiny apps for step-by-step instructions.

- {% elif app.slug == 'tissuumaps' %} -

This form allows you to start hosting your own TissUUmaps instance at SciLifeLab Serve. After your app has been created you will need to upload the data to your project. We suggest that you create the app with Permissions initially set to private, upload the data, and then edit the app settings to change the Permissions to public. Please read our documentation page on TissUUmaps apps for step-by-step instructions.

- {% endif %} - {% if app.slug in 'jupyter-lab,rstudio,vscode' %} -

Note that after 7 days the created {{ app.name }} instance will be deleted, only the files saved in 'project-vol' will stay available.

-

Each {{ app.name }} instance can get access to the persistent volume (folder) associated with this project, called 'project-vol'. Please make sure to save all your data files, script files, output from computations, etc. inside 'project-vol'; the files located elsewhere can be deleted at any point. The files saved inside 'project-vol' will be available across all instances of {{ app.name }} (as well as other apps) within this project.

- - {% endif %} -
-
-
-
-
- -
- {% csrf_token %} - - {% if app.slug == "filemanager" %} - - -
-

You are about to activate file manager on SciLifeLab Serve. You can use it to upload or download files to a volume associated with this project. This service will be active for 24 hours and automatically terminated afterwards. The uploaded files will stay on the volume even after this service has been terminated.

-

Click 'Activate' to activate file manager

-
-
- {% else %} -
- {% endif %} - - - -
- - - - -
- Please add a valid name! -
-
- {% if do_display_description_field %} -
- - - - -
- Please add a description of the app! -
-
- {% endif %} - - {% if do_display_description_field or request.user.is_superuser %} -
- - - -
- - -
-
- - {% endif %} - - - {% if form.dep_permissions %} -
- - - - - -
- {% endif %} - - {% if app.slug in 'customapp,dashapp,shinyapp,shinyproxyapp' %} -
- - - - -
- Please add a valid URL! -
-
- {% endif %} - - {% if form.dep_model %} -
- - - - -
- {% endif %} - - {% if form.dep_S3 %} -
- - -
- {% endif %} - - {% if form.dep_flavor %} -
- - - - -
- {% endif %} - - {% if form.dep_environment %} - {% if form.environments.objs.count > 0 %} -
- - -
- {% endif %} - {% endif %} - - {% if form.dep_apps %} - {% for app_name, appinstances in form.app_deps.items %} -
- - -
- {% endfor %} - {% endif %} - - {% for key, vals in form.primitives.items %} - -
{{ vals.meta_title }}
- {% for subkey, subval in vals.items %} - {% if subval.type != "boolean" and subkey == "image" or subkey == "port" or subkey == "userid" or subkey == "proxyheartbeatrate" or subkey == "proxyheartbeattimeout" or subkey == "proxycontainerwaittime" %} -
- {% else %} -
- {% endif %} - {% if subval.type == "boolean" %} - - - {% endif %} - - {% if subval.type == "string" %} - - {% if subkey == "image" %} - - - -
- Please add a valid image name! -
- {% elif subkey == "path" %} - - - {% if request.user.is_superuser %} - - -
- Please add a valid path! -
- {% else %} -
- /home/ - -
- Please add a valid path! -
-
- {% endif %} - - {% else %} - - {% endif %} - {% endif %} - - {% if subval.type == "password" %} - - - {% endif %} - - {% if subval.type == "textfield" %} - - - {% endif %} - - {% if subval.type == "number" %} - {% if subkey == "port" or subkey == "userid" or subkey == "proxyheartbeatrate" or subkey == "proxyheartbeattimeout" or subkey == "proxycontainerwaittime" %} - {% if subkey == "port" %} - - - - -
- Please add a valid port! -
- {% elif subkey == "userid" and request.user.is_superuser %} - - - - -
- Please add a valid User ID between 999 and 1010! -
- {% elif subkey == "proxyheartbeatrate" and request.user.is_superuser %} - - - - -
- Please add a number, needs to be above 1. -
- {% elif subkey == "proxyheartbeattimeout" and request.user.is_superuser %} - - - - -
- Please add a number. Cannot be lower than 0 except for '-1' to set no timeout. -
- {% elif subkey == "proxycontainerwaittime" and request.user.is_superuser %} - - - - -
- Please add a number, needs to be above 20000. -
- {% endif %} - {% else %} - - {% endif %} - {% endif %} - - {% if subval.type == "select" %} - - - - - - {% endif %} - -
- {% endfor %} - - {% endfor %} - - {% if form.dep_appobj %} -
- - -
- {% endif %} -
- - - - -
-
-
- -{# In the custom app creation form, make the path a required field if a persistent volumet is selected #} -{% if app.slug == "customapp" %} - -{% endif %} - - - -{% endblock %} diff --git a/templates/apps/create_base.html b/templates/apps/create_base.html new file mode 100644 index 000000000..a9e7f236e --- /dev/null +++ b/templates/apps/create_base.html @@ -0,0 +1,186 @@ +{% extends 'base.html' %} +{% block title %}Create {{ app.name }}{% endblock %} +{% load static %} +{% load custom_tags %} +{% load crispy_forms_tags %} + + +{% block content %} + +{% include "breadcrumbs/bc_app_create.html" %} + +
+
+
+
+ {% if app_id %} +

Edit {{ form.instance.name }}

+ {% else %} +

Create {{ form.model_name }}

+

+ {% block app_info %} + {% endblock %} +

+ {% endif %} + +
+ {% crispy form %} +
+
+
+
+ + + + +{% endblock %} diff --git a/templates/apps/create_view.html b/templates/apps/create_view.html new file mode 100644 index 000000000..4a30b1309 --- /dev/null +++ b/templates/apps/create_view.html @@ -0,0 +1,19 @@ +{% extends "apps/create_base.html" %} + +{% block app_info %} + {% if app_slug == 'customapp' %} +

This form allows you to start hosting a custom app that fulfills certain requirements on SciLifeLab Serve (the app itself can be built on any framework). Please read our documentation page on custom apps for the list of requirements and step-by-step instructions on deployment.

+ {% elif app_slug == 'dashapp' %} +

This form allows you to start hosting a Dash app at SciLifeLab Serve. Please read our documentation page on Dash apps for step-by-step instructions.

+ {% elif app_slug in 'shinyapp,shinyproxyapp' %} +

This form allows you to start hosting a Shiny app at SciLifeLab Serve. Please read our documentation page on Shiny apps for step-by-step instructions.

+ {% elif app_slug == 'tissuumaps' %} +

This form allows you to start hosting your own TissUUmaps instance at SciLifeLab Serve. After your app has been created you will need to upload the data to your project. We suggest that you create the app with Permissions initially set to private, upload the data, and then edit the app settings to change the Permissions to public. Please read our documentation page on TissUUmaps apps for step-by-step instructions.

+ {% endif %} + {% if app_slug in 'jupyter-lab,rstudio,vscode' %} +

Note that after 7 days the created {{ form.model_name }} instance will be deleted, only the files saved in 'project-vol' will stay available.

+

Each {{ form.model_name }} instance can get access to the persistent volume (folder) associated with this project, called 'project-vol'. Please make sure to save all your data files, script files, output from computations, etc. inside 'project-vol'; the files located elsewhere can be deleted at any point. The files saved inside 'project-vol' will be available across all instances of {{ form.model_name }} (as well as other apps) within this project.

+ + {% endif %} + +{% endblock%} diff --git a/templates/apps/custom_field.html b/templates/apps/custom_field.html new file mode 100644 index 000000000..a296c1dda --- /dev/null +++ b/templates/apps/custom_field.html @@ -0,0 +1,25 @@ +
+
+ + + +
+ +
+ {% if spinner %} +
+ +
+ {% endif %} + {{ field }} + + {% if field.help_text %} + {{ field.help_text }} + {% endif %} + {% for error in field.errors %} +
{{ error }}
+ {% endfor %} +
+
diff --git a/templates/apps/logs.html b/templates/apps/logs.html index 1e17ea3a0..55783c3c8 100644 --- a/templates/apps/logs.html +++ b/templates/apps/logs.html @@ -1,31 +1,25 @@ {% extends 'base.html' %} -{% block title %}Logs - {{ app.name }}{% endblock %} +{% block title %}Logs - {{ instance.name }}{% endblock %} {% load static %} {% block content %} - +{% include "breadcrumbs/bc_logs.html" %}
-

{{ app.name }} Logs

{{ app.status.latest.status_type }}
+

{{ instance.name }} Logs

{{ instance.status.latest.status_type }}
-

Note: Logs appear a few minutes after an app has been launched. The Status on the top right is an indication of the state of the app. {% if .sluapp.appg == 'customapp' and app.app_dependencies.all %} If the app is not running (due to an error) and you have a volume attached to the app, then you can switch between to the tabs below to see logs for the data copy process. This can give you hints if data copy failed.{% endif %}

+

Note: Logs appear a few minutes after an app has been launched. The Status on the top right is an indication of the state of the app. {% if instance.app.slug == 'customapp' and instance.volume %} If the app is not running (due to an error) and you have a volume attached to the app, then you can switch between to the tabs below to see logs for the data copy process. This can give you hints if data copy failed.{% endif %}

@@ -46,10 +40,9 @@

{{ app.name }} Logs

{{ app.name }} Logs
- - - -

Settings {{ app.name }}

-
-
- {% if app.slug == 'customapp' %} -

This form allows you to host an app that fulfills certain requirements on SciLifeLab Serve (the app itself can be built on any framework). Please read our documentation page on custom apps for the list of requirements and step-by-step instructions on deployment.

- {% elif app.slug == 'dashapp' %} -

This form allows you to host a Dash app at SciLifeLab Serve. Please read our documentation page on Dash apps for step-by-step instructions.

- {% elif app.slug in 'shinyapp,shinyproxyapp' %} -

This form allows you to host a Shiny app at SciLifeLab Serve. Please read our documentation page on Shiny apps for step-by-step instructions.

- {% elif app.slug == 'tissuumaps' %} -

You are editing a TissUUmaps app. Remember to upload the data to your project and then edit the app settings to change the Permissions to public. Please read our documentation page on TissUUmaps apps for step-by-step instructions.

- {% endif %} - {% if app.slug in 'jupyter-lab,rstudio,vscode' %} -

Note that after 7 days the created {{ app.name }} instance will be deleted, only the files saved in 'project-vol' will stay available.

-

Each {{ app.name }} instance can get access to the persistent volume (folder) associated with this project, called 'project-vol'. Please make sure to save all your data files, script files, output from computations, etc. inside 'project-vol'; the files located elsewhere can be deleted at any point. The files saved inside 'project-vol' will be available across all instances of {{ app.name }} (as well as other apps) within this project.

- {% endif %} -
-
- -
-
-
-
- {% csrf_token %} -
- -
- - - - -
- Please add a valid name! -
-
- - {% if do_display_description_field %} -
- - - - -
- Please add a description of the app! -
-
- {% endif %} - {% if do_display_description_field or request.user.is_superuser %} -
- - - -
- - -
-
- - {% endif %} - {% if form.dep_permissions %} -
- - - - - - -
- {% endif %} - - {% if app.slug in 'customapp,dashapp,shinyapp,shinyproxyapp' %} -
- - - - -
- Please add a valid URL! -
-
- {% endif %} - - {% if form.dep_model %} -
- - - - -
- {% endif %} - - {% if form.dep_S3 %} -
- - -
- {% endif %} - - {% if form.dep_flavor %} -
- - - - -
- {% endif %} - - {% if form.dep_environment %} - {% if form.environments.objs.count > 0 %} -
- - -
- {% endif %} - {% endif %} - - {% if form.dep_apps %} - {% for app_name, appinstances in form.app_deps.items %} -
- - -
- {% endfor %} - {% endif %} - - - {% for key, vals in form.primitives.items %} - -
{{ vals.meta_title }}
- {% for subkey, subval in vals.items %} - {% if subval.type != "boolean" and subkey == "image" or subkey == "port" or subkey == "userid" or subkey == "proxyheartbeatrate" or subkey == "proxyheartbeattimeout" or subkey == "proxycontainerwaittime" %} -
- {% else %} -
- {% endif %} - {% if subval.type == "boolean" %} - - - {% endif %} - - {% if subval.type == "string" %} - - {% if subkey == "image" %} - - - -
- Please add a valid image name! -
- {% elif subkey == "path" %} - - - {% if request.user.is_superuser and created_by_admin %} - -
- Please add a valid path! -
- {% elif created_by_admin %} -
- -
- {% else %} -
- /home/ - -
- Please add a valid path! -
-
- {% endif %} - - {% else %} - - {% endif %} - {% endif %} - - {% if subval.type == "textfield" %} - - - {% endif %} - - {% if subval.type == "password" %} - - -
- - - - -
- {% endif %} - - {% if subval.type == "number" %} - {% if subkey == "port" or subkey == "userid" or subkey == "proxyheartbeatrate" or subkey == "proxyheartbeattimeout" or subkey == "proxycontainerwaittime" %} - {% if subkey == "port" %} - - - - -
- Please add a valid port! -
- {% elif subkey == "userid" and request.user.is_superuser %} - - - - -
- Please add a valid User ID between 999 and 1010! -
- {% elif subkey == "proxyheartbeatrate" and request.user.is_superuser %} - - - - -
- Please add a number, needs to be above 1. -
- {% elif subkey == "proxyheartbeattimeout" and request.user.is_superuser %} - - - - -
- Please add a number. Cannot be lower than 0 except for '-1' to set no timeout. -
- {% elif subkey == "proxycontainerwaittime" and request.user.is_superuser %} - - - - -
- Please add a number, needs to be above 20000. -
- {% endif %} - {% else %} - - {% endif %} - {% endif %} - - - {% if subval.type == "select" %} - - - - - - {% endif %} - - {% if subval.type == "minio-username" %} - - - {% endif %} - - {% if subval.type == "minio-password" %} - - - {% endif %} - -
- {% endfor %} - - {% endfor %} - - {% if form.dep_appobj %} -
- - -
- {% endif %} -
- - -
-
-
- -{% if app.category.pk == 'serve' %} -
- -
- -
-

Tags

-

Add relevant keywords to help users find your app in our catalog of public apps.

- -
- - {% csrf_token %} - -
-
- - {% include 'common/autocomplete.html' with str_list=all_tags id_suffix="tags" name="tag" required=True %} - -
-
- -
-
- -
-
-
- -
- -
- -
- {% with appinstance.tags|split:"," as tags %} - {% for tag in tags %} -
-
- {% csrf_token %} - - {{tag}} - -
-
- {% endfor %} - - {% endwith %} -
-
-
-
-{% endif %} -
- -{% if app.slug == "customapp" %} - -{% endif %} - - -{% endblock %} diff --git a/templates/breadcrumbs/bc_app_create.html b/templates/breadcrumbs/bc_app_create.html new file mode 100644 index 000000000..eaf7b4907 --- /dev/null +++ b/templates/breadcrumbs/bc_app_create.html @@ -0,0 +1,10 @@ +{% extends "breadcrumbs/breadcrumb_base.html" %} + +{% block breadcrumb_content %} +{% if app_id %} + +{% else %} + +{% endif %} + +{% endblock %} diff --git a/templates/breadcrumbs/bc_logs.html b/templates/breadcrumbs/bc_logs.html new file mode 100644 index 000000000..00641fa54 --- /dev/null +++ b/templates/breadcrumbs/bc_logs.html @@ -0,0 +1,5 @@ +{% extends "breadcrumbs/breadcrumb_base.html" %} + +{% block breadcrumb_content %} + +{% endblock %} diff --git a/templates/breadcrumbs/bc_project_create.html b/templates/breadcrumbs/bc_project_create.html new file mode 100644 index 000000000..724eb1296 --- /dev/null +++ b/templates/breadcrumbs/bc_project_create.html @@ -0,0 +1,6 @@ +{% extends "breadcrumbs/breadcrumb_base.html" %} + +{% block breadcrumb_content %} + + +{% endblock %} diff --git a/templates/breadcrumbs/bc_project_overview.html b/templates/breadcrumbs/bc_project_overview.html new file mode 100644 index 000000000..eeb5beafd --- /dev/null +++ b/templates/breadcrumbs/bc_project_overview.html @@ -0,0 +1 @@ +{% extends "breadcrumbs/breadcrumb_base.html" %} diff --git a/templates/breadcrumbs/breadcrumb_base.html b/templates/breadcrumbs/breadcrumb_base.html new file mode 100644 index 000000000..5ccc92b87 --- /dev/null +++ b/templates/breadcrumbs/breadcrumb_base.html @@ -0,0 +1,10 @@ + diff --git a/templates/collections/collection.html b/templates/collections/collection.html index 65799bc2b..bb51944dc 100644 --- a/templates/collections/collection.html +++ b/templates/collections/collection.html @@ -8,7 +8,7 @@ diff --git a/templates/common/app_card copy.html b/templates/common/app_card copy.html new file mode 100644 index 000000000..aaa38ea31 --- /dev/null +++ b/templates/common/app_card copy.html @@ -0,0 +1,119 @@ +{% load static %} +{% load custom_tags %} + +
+
+
+
+
{{ app.name }}
+
+
+ {% static 'images/logos/apps/' as static_url %} + App Logo +
+
+ +
+

+ {% if app.description|length > 349 %} + {{ app.description|slice:':349'}}... + + {% else %} + {{ app.description|default_if_none:""}} + {% endif %} +

+
    +
  • +
    Owner:
    +
    {{ app.owner.first_name }} {{ app.owner.last_name }}
    +
  • +
+
+ +
+
+ {% if request.session.app_tags|exists:app.id %} + {% with app.tags as tags %} + {% for tag in tags %} + +
{{ tag }} +
+ {% endfor %} + + {% csrf_token %} + + + {% endwith %} + {% else %} + {% with app.tags as tags %} + + {% for tag in tags %} + +
{{ tag }} +
+ {% endfor %} + {% if tags.count > tag_limit %} + + {% csrf_token %} + + + {% endif %} + {% endwith %} + + {% endif %} +
+
+
+
+ {% if "Serve" in app.app.name or app.app.name == "Python Model Deployment" %} + Copy API Endpoint + {% else %} + Open + {% endif %} + {% if app.app.slug in 'shinyapp,shinyproxyapp,dashapp,customapp' %} + {% if app.pvc == None %} + + + + + + {% endif %} + {% endif %} + {% if app.source_code_url %} + + + + + + + {% endif %} +
+
+ {% if app.status_group == "success" %} + Running + {% else %} + Waiting + {% endif %} +
+
+ +
+
+ + +
diff --git a/templates/common/app_card.html b/templates/common/app_card.html index 2884200de..31cf09413 100644 --- a/templates/common/app_card.html +++ b/templates/common/app_card.html @@ -35,7 +35,7 @@
{{ app.name }}
{% if request.session.app_tags|exists:app.id %} - {% with app.tags|split:"," as tags %} + {% with app.tags.all as tags %} {% for tag in tags %} @@ -50,7 +50,7 @@
{{ app.name }}
{% endwith %} {% else %} - {% with app.tags|split:"," as tags %} + {% with app.tags.all as tags %} {% with tags|count_str as tag_limit %} {% for tag in tags|slice:tag_limit %} {{ app.name }}
{% if "Serve" in app.app.name or app.app.name == "Python Model Deployment" %} - Copy API Endpoint + Copy API Endpoint {% else %} - Open + Open {% endif %} {% if app.app.slug in 'shinyapp,shinyproxyapp,dashapp,customapp' %} {% if app.pvc == None %} diff --git a/templates/common/footer.html b/templates/common/footer.html index d5c942e4b..c1aee30ff 100644 --- a/templates/common/footer.html +++ b/templates/common/footer.html @@ -24,7 +24,7 @@
  • Home
  • Public apps
  • Public models
  • -
  • Collections
  • +
  • Collections
  • User guide
  • Teaching
  • About
  • diff --git a/templates/portal/home.html b/templates/portal/home.html index 1ba7e7ea9..7e5072513 100644 --- a/templates/portal/home.html +++ b/templates/portal/home.html @@ -168,7 +168,7 @@

    Collections {% if link_all_collections %} {% endif %}

    @@ -187,7 +187,7 @@

    News
    - +
    diff --git a/templates/projects/categories/develop.html b/templates/projects/categories/develop.html new file mode 100644 index 000000000..d121a40d4 --- /dev/null +++ b/templates/projects/categories/develop.html @@ -0,0 +1,13 @@ +{% extends 'projects/partials/category_card_base.html' %} + +{% block top_content %} + +{% endblock %} + +{% block bottom_content %} + +
    +*Note that all apps under Develop will be deleted 7 days after creation. +
    + +{% endblock%} diff --git a/templates/projects/categories/manage_files copy.html b/templates/projects/categories/manage_files copy.html new file mode 100644 index 000000000..458b81a78 --- /dev/null +++ b/templates/projects/categories/manage_files copy.html @@ -0,0 +1,61 @@ +
    +
    +
    +
    Manage Files
    +
    + {% if filemanager_instance %} + Status: + + {{ filemanager_instance.status.latest.status_type }} + + {% endif %} +
    +
    + {% if filemanager_instance %} +
    +
    + +

    File Manager is activated and will be closed in {{ hours }} hours

    +

    You can reach the manager by clicking the button below

    + + Open File Manager + +
    + +
    + {% else %} + + +
    +
    +
    +
    +
    +
    +
    +
    Activate File Manager
    +
    +
    +
    + {% if filemanager_instance %} + + {% else %} + Activate + {% endif %} + +
    +
    +
    +
    + App Logo +
    +
    +
    +
    +
    +
    + {% endif %} +
    +
    diff --git a/templates/projects/categories/manage_files.html b/templates/projects/categories/manage_files.html new file mode 100644 index 000000000..4e8acb272 --- /dev/null +++ b/templates/projects/categories/manage_files.html @@ -0,0 +1,13 @@ +{% extends 'projects/partials/category_card_base.html' %} + +{% block top_content %} + +{% endblock %} + +{% block bottom_content %} + +
    +*Note that the file manager will be deleted after 24 hours. +
    + +{% endblock%} diff --git a/templates/projects/categories/models.html b/templates/projects/categories/models.html new file mode 100644 index 000000000..9e89448a2 --- /dev/null +++ b/templates/projects/categories/models.html @@ -0,0 +1,137 @@ + +{% if project.project_template.pk == 1 %} +
    +
    +
    +
    Models
    +
    + {% if models %} + +
    + +{% endif %} diff --git a/templates/projects/categories/serve.html b/templates/projects/categories/serve.html new file mode 100644 index 000000000..81369f5aa --- /dev/null +++ b/templates/projects/categories/serve.html @@ -0,0 +1,9 @@ +{% extends 'projects/partials/category_card_base.html' %} + +{% block content %} + +
    +*Add specifics here +
    + +{% endblock%} diff --git a/templates/projects/overview.html b/templates/projects/overview.html index 05c2c14f3..d85ba12a7 100644 --- a/templates/projects/overview.html +++ b/templates/projects/overview.html @@ -1,6 +1,7 @@ {% extends 'base.html' %} + {% load static %} -{% load can_create_app %} + {% load custom_tags %} {% block title %}{{ project.name }}{% endblock %} @@ -8,575 +9,37 @@ {% if project.status == "active" %} - - -
    -

    {{ project.name }}

    - - - Settings - -
    - -
    -
    - {% include 'common/flash_messages.html' %} -
    -
    - -
    -
    - {% if project.description %} -

    Description: {{ project.description }}

    - {% endif %} -

    Project owner: {{ project.owner.email }}

    - {% if project.authorized.all %} -

    Project members: {{ project.authorized.all|join:"; " }}

    - {% endif %} -
    -
    - - - -
    - {% for objs in resources %} - {% if objs.title == "Manage Files" and not request.user.is_superuser or objs.title == "Additional options [admins only]" and not request.user.is_superuser %} - {% else %} -
    -
    -
    -
    {{ objs.title }}
    -
    - {% if objs.objs %} - - - {% if objs.title == "Develop" %} -
    - *Note that all apps under Develop will be deleted 7 days after creation. -
    - {% endif %} - {% else %} -
    -

    No instances.

    -
    + {% if objs.title == "Develop" and objs.apps %} + {% include "projects/categories/develop.html" %} {% endif %} -
    - {% for app in objs.apps %} - {% if project.project_template.pk == 1 and app.slug in 'mlflow,mlflow-serve,pytorch-serve,tensorflow-serve,python-serve' %} - - {% elif app.slug == "shinyapp" and not request.user.is_superuser %} - - {% else %} -
    -
    -
    -
    -
    -
    -
    {{ app.name }}
    -
    -
    -
    -
    -
    -

    {{ app.description }}

    -
    -
    -
    -
    - {% can_create_app request.user project app as can_create %} - - {% if can_create %} - - - {% if "Serv" in app.name or app.name == "Python Model Deployment" %} - Create - {% else %} - - Create - - - {% endif %} - - - {% else %} - - {% endif %} -
    -
    -
    -
    -
    -
    - {% static 'images/logos/apps/' as static_url %} - App Logo -
    -
    -
    - {% if "Serv" in app.name or app.name == "Python Model Deployment" %} -
    -

    {{ app.name }} will be available soon

    -
    - {% endif %} -
    -
    - {% endif %} - {% endfor %} -
    -
    -
    - {% endif %} - {% endfor %} - - {% if project.project_template.pk == 2 %} -
    -
    -
    -
    Models
    -
    - {% if models %} - -
    - - {% endif %} - - - - -
    -
    -
    -
    Manage Files
    -
    - {% if filemanager_instance %} - Status: - - {{ filemanager_instance.status.latest.status_type }} - - {% endif %} -
    -
    - {% if filemanager_instance %} -
    -
    - -

    File Manager is activated and will be closed in {{ hours }} hours

    -

    You can reach the manager by clicking the button below

    - - Open File Manager - -
    - -
    - {% else %} - -
    -
    -
    -
    -
    -
    -
    -
    Activate File Manager
    -
    -
    -
    - {% if filemanager_instance %} - - {% else %} - Activate - {% endif %} - -
    -
    -
    -
    - App Logo -
    -
    -
    -
    -
    -
    + {% if objs.title == "Serve" and objs.apps %} + {% include "projects/categories/serve.html" %} {% endif %} -
    -
    -
    - - -
    - {% else %} + {% if objs.title == "Manage Files" and objs.apps %} + {% include "projects/categories/manage_files.html" %} + {% endif %} -
    -
    + {% if objs.title == "Additional options [admins only]" and objs.apps %} + {% include "projects/categories/manage_files.html" %} + {% endif %} + {% endfor %} - {% include 'common/loader.html' %} -
    +{% endif %} - - - {% endif %} - - - +{% include "projects/partials/scripts.html" %} - {% endblock %} +{% endblock %} diff --git a/templates/projects/partials/app_instances_table.html b/templates/projects/partials/app_instances_table.html new file mode 100644 index 000000000..62e020846 --- /dev/null +++ b/templates/projects/partials/app_instances_table.html @@ -0,0 +1,114 @@ +{% load static %} + + + + + + + + + + + + + + + {% for instance in objs.instances %} + + + {% static 'images/logos/apps/' as static_url %} + + + {% if instance.url %} + + {% else %} + + {% endif %} + + + + + + + {% endfor %} + + diff --git a/templates/projects/partials/app_templates.html b/templates/projects/partials/app_templates.html new file mode 100644 index 000000000..fddc7f4e9 --- /dev/null +++ b/templates/projects/partials/app_templates.html @@ -0,0 +1,68 @@ +{% load can_create_app %} +{% load static %} + +
    + {% for app in objs.apps %} + +
    +
    +
    +
    +
    +
    +
    {{ app.name }}
    +
    +
    +
    +
    +
    +

    {{ app.description }}

    +
    +
    +
    +
    + {% can_create_app request.user project app as can_create %} + + {% if can_create %} + + + {% if "Serv" in app.name or app.name == "Python Model Deployment" %} + Create + {% else %} + + Create + + + {% endif %} + + + {% else %} + + {% endif %} +
    +
    +
    +
    +
    +
    + {% static 'images/logos/apps/' as static_url %} + App Logo +
    +
    +
    + {% if "Serv" in app.name or app.name == "Python Model Deployment" %} +
    +

    {{ app.name }} will be available soon

    +
    + {% endif %} +
    +
    + {% endfor %} +
    diff --git a/templates/projects/partials/category_card_base.html b/templates/projects/partials/category_card_base.html new file mode 100644 index 000000000..e35225bb2 --- /dev/null +++ b/templates/projects/partials/category_card_base.html @@ -0,0 +1,28 @@ +
    +
    +
    +
    {{ objs.title }}
    +
    + + {% block top_content %} + + {% endblock %} + + {% if objs.instances %} + + + {% block bottom_content %} + {% endblock %} + + {% else %} +
    +

    No instances.

    +
    + {% endif %} + + {% include "projects/partials/app_templates.html" %} +
    +
    diff --git a/templates/projects/partials/project_description.html b/templates/projects/partials/project_description.html new file mode 100644 index 000000000..7cf2bacf7 --- /dev/null +++ b/templates/projects/partials/project_description.html @@ -0,0 +1,11 @@ +
    +
    + {% if project.description %} +

    Description: {{ project.description }}

    + {% endif %} +

    Project owner: {{ project.owner.email }}

    + {% if project.authorized.all %} +

    Project members: {{ project.authorized.all|join:"; " }}

    + {% endif %} +
    +
    diff --git a/templates/projects/partials/project_header.html b/templates/projects/partials/project_header.html new file mode 100644 index 000000000..15c944f45 --- /dev/null +++ b/templates/projects/partials/project_header.html @@ -0,0 +1,7 @@ +
    +

    {{ project.name }}

    + + + Settings + +
    diff --git a/templates/projects/partials/scripts.html b/templates/projects/partials/scripts.html new file mode 100644 index 000000000..27f34066b --- /dev/null +++ b/templates/projects/partials/scripts.html @@ -0,0 +1,118 @@ +{% if project.status == "active" %} + + +
    +{% else %} + +
    +
    + + {% include 'common/loader.html' %} +
    +
    + + + +{% endif %} diff --git a/templates/projects/project_create.html b/templates/projects/project_create.html index 6fca6fb99..f02e01d15 100644 --- a/templates/projects/project_create.html +++ b/templates/projects/project_create.html @@ -3,12 +3,7 @@ {% block title %}New project{% endblock %} {% block content %} - +{% include "breadcrumbs/bc_project_create.html" %}
    diff --git a/templates/projects/settings.html b/templates/projects/settings.html index caa45fc71..69e10861d 100644 --- a/templates/projects/settings.html +++ b/templates/projects/settings.html @@ -32,14 +32,6 @@

    Project settings

    Access {% if enable_extra_settings or request.user.is_superuser %} - - S3 storage - - - MLFlow - Flavors @@ -220,69 +212,8 @@
    Environments
    -
    -
    - -
    -
    Default S3 storage
    -
    - -
    -
    - {% csrf_token %} -
    - - -
    - -
    -
    -
    -
    - -
    -
    - -
    -
    Default MLFlow Server
    -
    - -
    -
    - {% csrf_token %} - -
    - - -
    - - -
    -
    -
    -
    {% endif %} {% if request.user.pk == project.owner.pk or request.user.is_superuser %}
    From e30d79291f0a8ebac99da519b4081f8a53f4615f Mon Sep 17 00:00:00 2001 From: Arnold Kochari Date: Tue, 2 Jul 2024 11:23:24 +0200 Subject: [PATCH 02/12] Project limit change + a few other small changes (#204) --- .../e2e/ui-tests/test-project-as-contributor.cy.js | 6 +++--- .../e2e/ui-tests/test-superuser-functionality.cy.js | 8 ++++---- fixtures/projects_templates.json | 6 +++--- studio/settings.py | 2 +- templates/apps/create_view.html | 2 +- templates/breadcrumbs/breadcrumb_base.html | 2 +- templates/projects/project_templates.html | 11 ++++++++--- 7 files changed, 21 insertions(+), 16 deletions(-) diff --git a/cypress/e2e/ui-tests/test-project-as-contributor.cy.js b/cypress/e2e/ui-tests/test-project-as-contributor.cy.js index 695948fad..5395cb589 100644 --- a/cypress/e2e/ui-tests/test-project-as-contributor.cy.js +++ b/cypress/e2e/ui-tests/test-project-as-contributor.cy.js @@ -257,8 +257,8 @@ describe("Test project contributor user functionality", () => { // Names of projects to create const project_name = "e2e-create-proj-test" - // Create 5 projects (current limit) - Cypress._.times(5, () => { + // Create 10 projects (current limit) + Cypress._.times(10, () => { cy.visit("/projects/") cy.get("a").contains('New project').click() cy.get("a").contains('Create').first().click() @@ -276,7 +276,7 @@ describe("Test project contributor user functionality", () => { cy.request({url: "/projects/templates/", failOnStatusCode: false}).its('status').should('equal', 403) // Now delete all created projects - Cypress._.times(5, () => { + Cypress._.times(10, () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('.confirm-delete').click() .then((href) => { diff --git a/cypress/e2e/ui-tests/test-superuser-functionality.cy.js b/cypress/e2e/ui-tests/test-superuser-functionality.cy.js index e093ca364..25874f5bb 100644 --- a/cypress/e2e/ui-tests/test-superuser-functionality.cy.js +++ b/cypress/e2e/ui-tests/test-superuser-functionality.cy.js @@ -290,8 +290,8 @@ describe("Test superuser access", () => { // Names of projects to create const project_name = "e2e-superuser-proj-limits-test" - cy.log("Create 5 projects (current limit for regular users)") - Cypress._.times(5, () => { + cy.log("Create 10 projects (current limit for regular users)") + Cypress._.times(10, () => { // better to write this out rather than use the createBlankProject command because then we can do a 5000 ms pause only once cy.visit("/projects/") cy.get("a").contains('New project').click() @@ -308,10 +308,10 @@ describe("Test superuser access", () => { cy.log("Create one more project to check it is possible to bypass the limit") cy.createBlankProject(project_name) cy.visit("/projects/") - cy.get('h5:contains("' + project_name + '")').its('length').should('eq', 6) // check that the superuser now bypassed the limit for regular users + cy.get('h5:contains("' + project_name + '")').its('length').should('eq', 11) // check that the superuser now bypassed the limit for regular users cy.log("Now delete all created projects") - Cypress._.times(6, () => { + Cypress._.times(11, () => { cy.visit("/projects/") cy.contains('.card-title', project_name).parents('.card-body').siblings('.card-footer').find('.confirm-delete').click() .then((href) => { diff --git a/fixtures/projects_templates.json b/fixtures/projects_templates.json index 6a3a27237..fc59f871c 100644 --- a/fixtures/projects_templates.json +++ b/fixtures/projects_templates.json @@ -1,7 +1,7 @@ [ { "fields": { - "description": "Use this template if you intend to only deploy apps or use integrated development environments (IDEs).", + "description": "Use this template if you intend to deploy apps or ML models packaged into containers, or if you intend to use web-based notebooks.", "name": "Default project", "slug": "blank", "available_apps": [8, 9, 19, 21, 22, 23, 24, 28], @@ -61,8 +61,8 @@ }, { "fields": { - "description": "Use this template if you intend to deploy machine learning models using the serving functionality.", - "name": "Project with ML serving", + "description": "This project type allows to deploy machine learning models using the specialized model serving frameworks.", + "name": "Project with specialized ML serving", "slug": "default", "template": { "apps": { diff --git a/studio/settings.py b/studio/settings.py index afecb5700..b9166f183 100644 --- a/studio/settings.py +++ b/studio/settings.py @@ -437,7 +437,7 @@ "filemanager": 1, } -PROJECTS_PER_USER_LIMIT = 5 +PROJECTS_PER_USER_LIMIT = 10 STUDIO_ACCESSMODE = os.environ.get("STUDIO_ACCESSMODE", "") ENABLE_PROJECT_EXTRA_SETTINGS = False diff --git a/templates/apps/create_view.html b/templates/apps/create_view.html index 4a30b1309..f2b52b42c 100644 --- a/templates/apps/create_view.html +++ b/templates/apps/create_view.html @@ -8,7 +8,7 @@ {% elif app_slug in 'shinyapp,shinyproxyapp' %}

    This form allows you to start hosting a Shiny app at SciLifeLab Serve. Please read our documentation page on Shiny apps for step-by-step instructions.

    {% elif app_slug == 'tissuumaps' %} -

    This form allows you to start hosting your own TissUUmaps instance at SciLifeLab Serve. After your app has been created you will need to upload the data to your project. We suggest that you create the app with Permissions initially set to private, upload the data, and then edit the app settings to change the Permissions to public. Please read our documentation page on TissUUmaps apps for step-by-step instructions.

    +

    This form allows you to start hosting your own TissUUmaps instance at SciLifeLab Serve. After your app has been created you will need to upload the data to your project. We suggest that you create the app with Permissions initially set to Private, upload the data, and then edit the app settings to change the Permissions to Public. Please read our documentation page on TissUUmaps apps for step-by-step instructions.

    {% endif %} {% if app_slug in 'jupyter-lab,rstudio,vscode' %}

    Note that after 7 days the created {{ form.model_name }} instance will be deleted, only the files saved in 'project-vol' will stay available.

    diff --git a/templates/breadcrumbs/breadcrumb_base.html b/templates/breadcrumbs/breadcrumb_base.html index 5ccc92b87..74f003b16 100644 --- a/templates/breadcrumbs/breadcrumb_base.html +++ b/templates/breadcrumbs/breadcrumb_base.html @@ -2,7 +2,7 @@