diff --git a/.github/workflows/python-app.yml b/.github/workflows/python-app.yml new file mode 100644 index 0000000..c5c532b --- /dev/null +++ b/.github/workflows/python-app.yml @@ -0,0 +1,33 @@ +name: mlex_latent_explorer + +on: + push: + branches: [ main ] + pull_request: + branches: [ main ] + +jobs: + test: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + - name: Set up Python 3.9 + uses: actions/setup-python@v5 + with: + python-version: 3.9 + cache: 'pip' + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install . + pip install .[dev] + - name: Lint with flake8 + run: | + # stop the build if there are Python syntax errors or undefined names + flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics + # exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide + flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics + - name: Test with pytest + run: | + pytest diff --git a/.gitignore b/.gitignore index 96318fb..0666eab 100644 --- a/.gitignore +++ b/.gitignore @@ -3,6 +3,7 @@ __pycache__/ *.py[cod] *$py.class test.py +**cache** # output dir results/ diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..4eef675 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,14 @@ +FROM python:3.9 +LABEL maintainer="THE MLEXCHANGE TEAM" + +RUN ls + +RUN pip3 install --upgrade pip &&\ + pip3 install -r . + +WORKDIR /app/work +ENV HOME /app/work +COPY src src + +CMD ["bash"] +CMD python3 src/frontend.py diff --git a/README.md b/README.md index 0b50cb3..895c29f 100644 --- a/README.md +++ b/README.md @@ -11,6 +11,35 @@ To run this demo: - clone this repo and then run `docker-compose up` in the main folder. - Go to `http://0.0.0.0:8070/` in a brower. +## Developer Setup +If you are developing this library, there are a few things to note. + +1. Install development dependencies: + +``` +pip install . +pip install ".[dev]" +``` + +2. Install pre-commit +This step will setup the pre-commit package. After this, commits will get run against flake8, black, isort. + +``` +pre-commit install +``` + +3. (Optional) If you want to check what pre-commit would do before commiting, you can run: + +``` +pre-commit run --all-files +``` + +4. To run test cases: + +``` +python -m pytest +``` + ## Copyright MLExchange Copyright (c) 2023, The Regents of the University of California, diff --git a/docker/Dockerfile b/docker/Dockerfile deleted file mode 100644 index 87d8d55..0000000 --- a/docker/Dockerfile +++ /dev/null @@ -1,17 +0,0 @@ -FROM python:3.9 -LABEL maintainer="THE MLEXCHANGE TEAM" - -RUN ls -COPY docker/requirements.txt requirements.txt - -RUN pip3 install --upgrade pip &&\ - pip3 install -r requirements.txt\ - pip install git+https://github.com/mlexchange/mlex_file_manager\ - pip install git+https://github.com/mlexchange/mlex_dash_component_editor - -WORKDIR /app/work -ENV HOME /app/work -COPY src src - -CMD ["bash"] -CMD python3 src/frontend.py diff --git a/docker/requirements.txt b/docker/requirements.txt deleted file mode 100644 index 1fda0c8..0000000 --- a/docker/requirements.txt +++ /dev/null @@ -1,16 +0,0 @@ -dash==2.9.3 -dash-core-components==2.0.0 -dash-bootstrap-components==1.0.2 -dash-html-components==2.0.0 -dash-iconify==0.1.2 -plotly==5.14.1 -scikit-learn==1.3.0 -dash-uploader==0.6.0 -requests==2.26.0 -pyarrow==11.0.0 -diskcache==5.6.3 -pandas -numpy -Pillow -python-dotenv -prefect-client==2.14.21 diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..99f17a1 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,52 @@ +[project] +name = "mlex_latent_explorer" +version = "0.0.1" +authors = [ + { name="Runbo Jiang", email="rjiang2@lbl.gov"} +] +description = "A web app to visualize latent vectors in 2D or 3D, supporting PCA and UMAP for dimension reduction." +readme = "README.md" +requires-python = ">=3.9" +classifiers = [ + "Programming Language :: Python :: 3", + "Operating System :: OS Independent", +] + +dependencies = [ + "dash==2.9.3", + "dash-core-components==2.0.0", + "dash-bootstrap-components==1.0.2", + "dash-html-components==2.0.0", + "dash-iconify==0.1.2", + "plotly==5.14.1", + "scikit-learn==1.3.0", + "dash-uploader==0.6.0", + "requests==2.26.0", + "pyarrow==11.0.0", + "diskcache==5.6.3", + "dash_component_editor@git+https://github.com/mlexchange/mlex_dash_component_editor", + "mlex_file_manager@git+https://github.com/mlexchange/mlex_file_manager", + "Pillow", + "pandas", + "numpy", + "python-dotenv", + "prefect-client==2.14.21" +] + +[project.optional-dependencies] +# These are required for developing the package (running the tests, building +# the documentation) but not necessarily required for _using_ it. +dev = [ + "dash[testing]", + "flake8", + "pre-commit", + "pytest-mock", + "tiled[all]", +] + +[project.urls] +Homepage = "https://github.com/mlexchange/mlex_latent_explorer" +Issues = "https://github.com/mlexchange/mlex_latent_explorer/issues/" + +[tool.isort] +profile = "black" diff --git a/src/__init__.py b/src/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/assets/__init__.py b/src/assets/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/frontend.py b/src/frontend.py index bc84a70..c8e6ca7 100755 --- a/src/frontend.py +++ b/src/frontend.py @@ -46,14 +46,14 @@ FLOW_TYPE = os.getenv("FLOW_TYPE", "podman") # Slurm -PARTITIONS_CPU = json.loads(os.getenv("PARTITIONS_CPU", [])) -RESERVATIONS_CPU = json.loads(os.getenv("RESERVATIONS_CPU", [])) +PARTITIONS_CPU = json.loads(os.getenv("PARTITIONS_CPU", "[]")) +RESERVATIONS_CPU = json.loads(os.getenv("RESERVATIONS_CPU", "[]")) MAX_TIME_CPU = os.getenv("MAX_TIME_CPU", "1:00:00") -PARTITIONS_GPU = json.loads(os.getenv("PARTITIONS_CPU", [])) -RESERVATIONS_GPU = json.loads(os.getenv("RESERVATIONS_CPU", [])) +PARTITIONS_GPU = json.loads(os.getenv("PARTITIONS_CPU", "[]")) +RESERVATIONS_GPU = json.loads(os.getenv("RESERVATIONS_CPU", "[]")) MAX_TIME_GPU = os.getenv("MAX_TIME_CPU", "1:00:00") SUBMISSION_SSH_KEY = os.getenv("SUBMISSION_SSH_KEY", "") -FORWARD_PORTS = json.loads(os.getenv("FORWARD_PORTS", [])) +FORWARD_PORTS = json.loads(os.getenv("FORWARD_PORTS", "[]")) # Mlex content api CONTENT_API_URL = os.getenv("CONTENT_API_URL", "http://localhost:8000/api/v0/models") diff --git a/src/test/__init__.py b/src/test/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/test/layout_test.py b/src/test/layout_test.py new file mode 100644 index 0000000..dd31977 --- /dev/null +++ b/src/test/layout_test.py @@ -0,0 +1,10 @@ +from src.frontend import show_clustering_gui_layouts + + +def test_clustering_layout(): + layout = show_clustering_gui_layouts("KMeans") + assert layout is not None + layout = show_clustering_gui_layouts("DBSCAN") + assert layout is not None + layout = show_clustering_gui_layouts("HDBSCAN") + assert layout is not None