diff --git a/Dockerfile b/Dockerfile index 2fb94e29e..c12929e9c 100644 --- a/Dockerfile +++ b/Dockerfile @@ -2,30 +2,55 @@ FROM python:3.9-slim ARG POETRY_VERSION="1.2.2" -WORKDIR /app +RUN groupadd --gid 1000 appuser \ + && useradd --uid 1000 --gid 1000 -ms /bin/bash appuser -COPY .env pyproject.toml poetry.lock README.md /app -COPY ./src/ /app/src - -ENV PYTHONPATH=${PYTHONPATH}:${PWD} +RUN pip3 install --no-cache-dir --upgrade \ + pip \ + virtualenv RUN apt-get update && apt-get install -y \ build-essential \ curl \ software-properties-common \ git \ - ffmpeg + ffmpeg \ + sqlite3 RUN apt-get -y clean \ && rm -rf /var/lib/apt/lists/* -RUN pip install . -WORKDIR /data +USER appuser +WORKDIR /home/appuser + +RUN curl -sSL https://install.python-poetry.org | \ + python - --version ${POETRY_VERSION} + +ENV PATH="/home/appuser/.local/bin:$PATH" + +COPY .env pyproject.toml poetry.lock /home/appuser + +RUN poetry cache clear pypi --all +RUN poetry config virtualenvs.create true \ + && poetry config virtualenvs.in-project true \ + && poetry config experimental.new-installer true \ + && poetry install --no-root --without dev + +COPY ./src/ /home/appuser/src +COPY README.md /home/appuser + +RUN poetry install --only-root EXPOSE 8501 -ENV STREAMLIT_SERVER_ADDRESS=0.0.0.0 +COPY run.sh /home/appuser + +USER root +RUN mkdir /data && chown appuser: /data -R +USER appuser + +RUN git config --global --add safe.directory '*' +EXPOSE 8000 -HEALTHCHECK CMD ecord-active --version -ENTRYPOINT ["encord-active"] +ENTRYPOINT ["./run.sh"] diff --git a/README.md b/README.md index e2b961cb5..9f26c5a4b 100644 --- a/README.md +++ b/README.md @@ -71,7 +71,7 @@ encord-active quickstart or you can use : ```shell -docker run -it --rm -p 8501:8501 -v ${PWD}:/data encord/encord-active quickstart +docker run -it --rm -p 8501:8501 -p 8000:8000 -v ${PWD}:/data encord/encord-active quickstart ``` After opening the UI, we recommend you to head to the [workflow documentation][encord-active-docs-workflow] to see some common workflows. diff --git a/docs/docs/getting-started.md b/docs/docs/getting-started.md index b9bfa2ff5..58dd39713 100644 --- a/docs/docs/getting-started.md +++ b/docs/docs/getting-started.md @@ -33,7 +33,7 @@ If you have trouble installing `encord-active`, you can find more detailed instr You can also run the docker image instead of install Encord Active ```shell -docker run -it -p 8501:8501 -v ${PWD}:/data encord/encord-active +docker run -it -p 8501:8501 -p 8000:8000 -v ${PWD}:/data encord/encord-active ``` ::: diff --git a/docs/docs/installation.mdx b/docs/docs/installation.mdx index b555420b8..c10004858 100644 --- a/docs/docs/installation.mdx +++ b/docs/docs/installation.mdx @@ -60,7 +60,7 @@ To learn more about how to use the command line interface, see the [Command Line We also provide a docker image which works exctly the same as the cli. ```shell -docker run -it --rm -p 8501:8501 -v ${PWD}:/data encord/encord-active +docker run -it --rm -p 8501:8501 -p 8000:8000 -v ${PWD}:/data encord/encord-active ``` This will mount your current working directory, so everything that happens inside the docker container will persist after it is done. @@ -70,7 +70,7 @@ This will mount your current working directory, so everything that happens insid If you intend to use Encord Active with an Encord Annotate project you'll need to mount a voulume with your SSH key as well. ```shell -docker run -it --rm -p 8501:8501 -v ${PWD}:/data -v ${HOME}/.ssh:/root/.ssh encord/encord-active +docker run -it --rm -p 8501:8501 -p 8000:8000 -v ${PWD}:/data -v ${HOME}/.ssh:/root/.ssh encord/encord-active ``` Then, when asked for your SSH key, you can point to `~/.ssh/` diff --git a/run.sh b/run.sh new file mode 100755 index 000000000..29bb9deb7 --- /dev/null +++ b/run.sh @@ -0,0 +1,27 @@ +#!/bin/bash + +APP_PID= +stopRunningProcess() { + # Based on https://linuxconfig.org/how-to-propagate-a-signal-to-child-processes-from-a-bash-script + if test ! "${APP_PID}" = '' && ps -p ${APP_PID} > /dev/null ; then + > /proc/1/fd/1 echo "Stopping ${COMMAND_PATH} which is running with process ID ${APP_PID}" + + kill -TERM ${APP_PID} + > /proc/1/fd/1 echo "Waiting for ${COMMAND_PATH} to process SIGTERM signal" + + wait ${APP_PID} + > /proc/1/fd/1 echo "All processes have stopped running" + else + > /proc/1/fd/1 echo "${COMMAND_PATH} was not started when the signal was sent or it has already been stopped" + fi +} + +trap stopRunningProcess EXIT TERM + +source .venv/bin/activate + +encord-active visualize --target /data + +APP_ID=${!} + +wait ${APP_ID} diff --git a/src/encord_active/server/main.py b/src/encord_active/server/main.py index feead19a5..28e220b3c 100644 --- a/src/encord_active/server/main.py +++ b/src/encord_active/server/main.py @@ -48,7 +48,7 @@ allow_headers=["*"], ) -app.mount("/static", StaticFiles(directory=path), name="static") +app.mount("/ea-static", StaticFiles(directory=path), name="static") async def get_project_file_structure(project: str) -> ProjectFileStructure: @@ -187,3 +187,8 @@ def search(project: ProjectFileStructureDep, query: str, type: SearchType, scope "ids": [item.identifier for item in result.result_identifiers], "snippet": snippet, } + + +@app.get("/health") +def health_check() -> bool: + return True diff --git a/src/encord_active/server/utils.py b/src/encord_active/server/utils.py index a0869ed0d..34ce3d250 100644 --- a/src/encord_active/server/utils.py +++ b/src/encord_active/server/utils.py @@ -49,7 +49,7 @@ def _get_url(label_row_structure: LabelRowStructure, du_hash: str, frame: str): label_row_structure.iter_data_unit(du_hash), None ) if data_unit: - return f"static/{parse.quote(data_unit.path.relative_to(label_row_structure.path.parents[2]).as_posix())}" + return f"ea-static/{parse.quote(data_unit.path.relative_to(label_row_structure.path.parents[2]).as_posix())}" def _transform_object(object_: dict):