diff --git a/.github/workflows/cypress-end-to-end-tests.yml b/.github/workflows/cypress-end-to-end-tests.yml
index 8f05430c9..4741b0113 100644
--- a/.github/workflows/cypress-end-to-end-tests.yml
+++ b/.github/workflows/cypress-end-to-end-tests.yml
@@ -379,47 +379,48 @@ jobs:
cd mephisto/scripts/local_db/gh_actions
python expire_all_units.py
- - name: 📚 Accepting the first submitted tip, accepting the second submitted tip, and rejecting the last submitted tip
- run: |
- cd mephisto/scripts/local_db
- python review_tips_for_task.py << EOF
- react-static-task-with-tips
- a
- 5
- The tip is very informative
- a
- 0
- r
- EOF
-
- - name: 🔥 Removing the second accepted tip
- run: |
- cd mephisto/scripts/local_db
- python remove_accepted_tip.py << EOF
- react-static-task-with-tips
- n
- y
- EOF
-
- - name: 📖 Reviewing the accepted feedback
- run: |
- cd mephisto/scripts/local_db
- python review_feedback_for_task.py << EOF
- react-static-task-with-tips
- 0
- n
- u
- y
- y
- EOF
- python review_feedback_for_task.py << EOF
- react-static-task-with-tips
- 1
- n
- u
- y
- n
- EOF
+# TODO: Fix tests for WorkerOpinion widget
+# - name: 📚 Accepting the first submitted tip, accepting the second submitted tip, and rejecting the last submitted tip
+# run: |
+# cd mephisto/scripts/local_db
+# python review_tips_for_task.py << EOF
+# react-static-task-with-tips
+# a
+# 5
+# The tip is very informative
+# a
+# 0
+# r
+# EOF
+#
+# - name: 🔥 Removing the second accepted tip
+# run: |
+# cd mephisto/scripts/local_db
+# python remove_accepted_tip.py << EOF
+# react-static-task-with-tips
+# n
+# y
+# EOF
+#
+# - name: 📖 Reviewing the accepted feedback
+# run: |
+# cd mephisto/scripts/local_db
+# python review_feedback_for_task.py << EOF
+# react-static-task-with-tips
+# 0
+# n
+# u
+# y
+# y
+# EOF
+# python review_feedback_for_task.py << EOF
+# react-static-task-with-tips
+# 1
+# n
+# u
+# y
+# n
+# EOF
- name: ⌛️ Running post-submission cypress tests
uses: cypress-io/github-action@v6.7.1
diff --git a/.github/workflows/docker-testing-matrix-manual-install.yml b/.github/workflows/docker-testing-matrix-manual-install.yml
index 31fca7abe..9df81cfce 100644
--- a/.github/workflows/docker-testing-matrix-manual-install.yml
+++ b/.github/workflows/docker-testing-matrix-manual-install.yml
@@ -41,6 +41,7 @@ jobs:
run: |
docker run mephisto_manual
- - name: Run command that removes and rebuilds all React apps related to the FormComposer
+ # Test that all Mephisto React apps can be built
+ - name: Run command that removes and rebuilds all Mephisto React apps
run: |
- docker run mephisto_manual bash -c 'mephisto scripts form_composer rebuild_all_apps'
+ docker run mephisto_manual bash -c 'mephisto scripts tests rebuild_all_mephisto_react_apps'
diff --git a/Dockerfile b/Dockerfile
index 542ff92bc..77f186e81 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,6 +1,6 @@
# Using the -slim version below for minimal size. You may want to
# remove -slim, or switch to -alpine if encountering issues
-ARG BASE_TAG=python3.9-nodejs16-slim
+ARG BASE_TAG=python3.9-nodejs22-slim
ARG BASE_IMAGE=nikolaik/python-nodejs:$BASE_TAG
FROM $BASE_IMAGE
diff --git a/docker/dockerfiles/Dockerfile.ubuntu-24.04 b/docker/dockerfiles/Dockerfile.ubuntu-24.04
index 8e450765b..2ff585a22 100644
--- a/docker/dockerfiles/Dockerfile.ubuntu-24.04
+++ b/docker/dockerfiles/Dockerfile.ubuntu-24.04
@@ -26,7 +26,7 @@ RUN curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.7/install.sh | b
RUN export NVM_DIR="$HOME/.nvm" \
&& [ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh" \
&& [ -s "$NVM_DIR/bash_completion" ] && \. "$NVM_DIR/bash_completion" \
- && nvm install 16 \
+ && nvm install 22 \
&& ln -s $(which node) /usr/bin/node \
&& ln -s $(which npm) /usr/bin/npm
@@ -43,6 +43,8 @@ COPY . $MEPHISTO_REPO_PATH
# Upgrade pip so we can use the `pyproject.toml` without raising an error
RUN pip install --upgrade pip
# Install Python requirements
+# [FOR DOCKERFILE ONLY] Requirements for Mephisto example `mnist`. Uncomment if you need them
+# RUN pip install torch pillow numpy detoxify
# [FOR DOCKERFILE ONLY] `--ignore-installed` - some libs can be preinstall in Docker-system
# Use `cd /mephisto && pip install -e .` in your local environment
RUN cd /mephisto && pip install --ignore-installed -e .
diff --git a/docs/web/docs/guides/how_to_use/efficiency_organization/manual_installation.md b/docs/web/docs/guides/how_to_use/efficiency_organization/manual_installation.md
index 2ff2e7c31..c4bfe2773 100644
--- a/docs/web/docs/guides/how_to_use/efficiency_organization/manual_installation.md
+++ b/docs/web/docs/guides/how_to_use/efficiency_organization/manual_installation.md
@@ -62,7 +62,7 @@ curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.7/install.sh | bash
export NVM_DIR="$HOME/.nvm" \
&& [ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh" \
&& [ -s "$NVM_DIR/bash_completion" ] && \. "$NVM_DIR/bash_completion" \
- && nvm install 16 \
+ && nvm install 22 \
&& ln -s $(which node) /usr/bin/node \
&& ln -s $(which npm) /usr/bin/npm
npm install -g yarn
diff --git a/examples/form_composer_demo/run_task.py b/examples/form_composer_demo/run_task.py
index a27617519..4169435e6 100644
--- a/examples/form_composer_demo/run_task.py
+++ b/examples/form_composer_demo/run_task.py
@@ -4,63 +4,21 @@
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.
-import os
-
from omegaconf import DictConfig
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
@task_script(default_config_file="example_local_mock")
def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
-
- operator.launch_task_run(cfg.mephisto)
- operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-
-
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:simple:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
+ examples.build_form_composer_simple(
force_rebuild=cfg.mephisto.task.force_rebuild,
post_install_script=cfg.mephisto.task.post_install_script,
- build_command="dev:simple",
)
+ operator.launch_task_run(cfg.mephisto)
+ operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
if __name__ == "__main__":
diff --git a/examples/form_composer_demo/run_task_dynamic.py b/examples/form_composer_demo/run_task_dynamic.py
index 06e8b0085..0baf14bc6 100644
--- a/examples/form_composer_demo/run_task_dynamic.py
+++ b/examples/form_composer_demo/run_task_dynamic.py
@@ -15,65 +15,12 @@
from mephisto.generators.form_composer.config_validation.task_data_config import (
create_extrapolated_config,
)
-from mephisto.generators.form_composer.config_validation.utils import set_custom_triggers_js_env_var
-from mephisto.generators.form_composer.config_validation.utils import (
- set_custom_validators_js_env_var,
-)
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-@task_script(default_config_file="dynamic_example_local_mock")
-def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
-
- operator.launch_task_run(cfg.mephisto)
- operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-
-
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
-
-
-def generate_task_data_json_config():
+def _generate_task_data_json_config():
"""
Generate extrapolated `task_data.json` config file,
based on existing form and tokens values config files
@@ -95,12 +42,17 @@ def generate_task_data_json_config():
data_path=data_path,
)
- # Set env var for `custom_validators.js`
- set_custom_validators_js_env_var(data_path)
- # Set env var for `custom_triggers.js`
- set_custom_triggers_js_env_var(data_path)
+
+@task_script(default_config_file="dynamic_example_local_mock")
+def main(operator: Operator, cfg: DictConfig) -> None:
+ examples.build_form_composer_dynamic(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
+ operator.launch_task_run(cfg.mephisto)
+ operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
if __name__ == "__main__":
- generate_task_data_json_config()
+ _generate_task_data_json_config()
main()
diff --git a/examples/form_composer_demo/run_task_dynamic_ec2_mturk_sandbox.py b/examples/form_composer_demo/run_task_dynamic_ec2_mturk_sandbox.py
index 415905ee0..edf9e92d5 100644
--- a/examples/form_composer_demo/run_task_dynamic_ec2_mturk_sandbox.py
+++ b/examples/form_composer_demo/run_task_dynamic_ec2_mturk_sandbox.py
@@ -21,79 +21,14 @@
from mephisto.generators.form_composer.config_validation.task_data_config import (
create_extrapolated_config,
)
-from mephisto.generators.form_composer.config_validation.utils import set_custom_triggers_js_env_var
-from mephisto.generators.form_composer.config_validation.utils import (
- set_custom_validators_js_env_var,
-)
from mephisto.generators.form_composer.constants import TOKEN_END_REGEX
from mephisto.generators.form_composer.constants import TOKEN_START_REGEX
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-@task_script(default_config_file="dynamic_example_ec2_mturk_sandbox")
-def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
-
- shared_state = SharedStaticTaskState()
-
- # Mephisto qualifications
- shared_state.qualifications = [
- # Custom Mephisto qualifications
- ]
-
- # Mturk qualifications
- shared_state.mturk_specific_qualifications = [
- # MTurk-specific quality control qualifications
- ]
-
- operator.launch_task_run(cfg.mephisto, shared_state)
- operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-
-
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
-
-
-def generate_data_json_config():
+def _generate_data_json_config():
"""
Generate extrapolated `task_data.json` config file,
based on existing form and tokens values config files
@@ -115,13 +50,8 @@ def generate_data_json_config():
data_path=data_path,
)
- # Set env var for `custom_validators.js`
- set_custom_validators_js_env_var(data_path)
- # Set env var for `custom_triggers.js`
- set_custom_triggers_js_env_var(data_path)
-
-def generate_preview_html():
+def _generate_preview_html():
"""
Generate HTML preview of a Task (based on first form version contained in `task_data.json`)
"""
@@ -168,7 +98,30 @@ def generate_preview_html():
f.write(preview_template)
+@task_script(default_config_file="dynamic_example_ec2_mturk_sandbox")
+def main(operator: Operator, cfg: DictConfig) -> None:
+ examples.build_form_composer_dynamic_ec2_mturk_sandbox(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
+
+ shared_state = SharedStaticTaskState()
+
+ # Mephisto qualifications
+ shared_state.qualifications = [
+ # Custom Mephisto qualifications
+ ]
+
+ # Mturk qualifications
+ shared_state.mturk_specific_qualifications = [
+ # MTurk-specific quality control qualifications
+ ]
+
+ operator.launch_task_run(cfg.mephisto, shared_state)
+ operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
+
+
if __name__ == "__main__":
- generate_data_json_config()
- generate_preview_html()
+ _generate_data_json_config()
+ _generate_preview_html()
main()
diff --git a/examples/form_composer_demo/run_task_dynamic_ec2_prolific.py b/examples/form_composer_demo/run_task_dynamic_ec2_prolific.py
index cc99edbe7..634af0e81 100644
--- a/examples/form_composer_demo/run_task_dynamic_ec2_prolific.py
+++ b/examples/form_composer_demo/run_task_dynamic_ec2_prolific.py
@@ -19,20 +19,41 @@
from mephisto.generators.form_composer.config_validation.task_data_config import (
create_extrapolated_config,
)
-from mephisto.generators.form_composer.config_validation.utils import set_custom_triggers_js_env_var
-from mephisto.generators.form_composer.config_validation.utils import (
- set_custom_validators_js_env_var,
-)
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
from mephisto.utils.qualifications import make_qualification_dict
+def _generate_data_json_config():
+ """
+ Generate extrapolated `task_data.json` config file,
+ based on existing form and tokens values config files
+ """
+ app_path = os.path.dirname(os.path.abspath(__file__))
+ data_path = os.path.join(app_path, FORM_COMPOSER__DATA_DIR_NAME, "dynamic")
+
+ form_config_path = os.path.join(data_path, FORM_COMPOSER__FORM_CONFIG_NAME)
+ token_sets_values_config_path = os.path.join(
+ data_path,
+ FORM_COMPOSER__TOKEN_SETS_VALUES_CONFIG_NAME,
+ )
+ task_data_config_path = os.path.join(data_path, FORM_COMPOSER__DATA_CONFIG_NAME)
+
+ create_extrapolated_config(
+ form_config_path=form_config_path,
+ token_sets_values_config_path=token_sets_values_config_path,
+ task_data_config_path=task_data_config_path,
+ data_path=data_path,
+ )
+
+
@task_script(default_config_file="dynamic_example_ec2_prolific")
def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
+ examples.build_form_composer_dynamic_ec2_prolific(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
shared_state = SharedStaticTaskState()
@@ -55,74 +76,6 @@ def main(operator: Operator, cfg: DictConfig) -> None:
operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
-
-
-def generate_data_json_config():
- """
- Generate extrapolated `task_data.json` config file,
- based on existing form and tokens values config files
- """
- app_path = os.path.dirname(os.path.abspath(__file__))
- data_path = os.path.join(app_path, FORM_COMPOSER__DATA_DIR_NAME, "dynamic")
-
- form_config_path = os.path.join(data_path, FORM_COMPOSER__FORM_CONFIG_NAME)
- token_sets_values_config_path = os.path.join(
- data_path,
- FORM_COMPOSER__TOKEN_SETS_VALUES_CONFIG_NAME,
- )
- task_data_config_path = os.path.join(data_path, FORM_COMPOSER__DATA_CONFIG_NAME)
-
- create_extrapolated_config(
- form_config_path=form_config_path,
- token_sets_values_config_path=token_sets_values_config_path,
- task_data_config_path=task_data_config_path,
- data_path=data_path,
- )
-
- # Set env var for `custom_validators.js`
- set_custom_validators_js_env_var(data_path)
- # Set env var for `custom_triggers.js`
- set_custom_triggers_js_env_var(data_path)
-
-
if __name__ == "__main__":
- generate_data_json_config()
+ _generate_data_json_config()
main()
diff --git a/examples/form_composer_demo/run_task_dynamic_presigned_urls_ec2_prolific.py b/examples/form_composer_demo/run_task_dynamic_presigned_urls_ec2_prolific.py
index 6595c7ec9..2c4aee6df 100644
--- a/examples/form_composer_demo/run_task_dynamic_presigned_urls_ec2_prolific.py
+++ b/examples/form_composer_demo/run_task_dynamic_presigned_urls_ec2_prolific.py
@@ -21,74 +21,11 @@
from mephisto.generators.form_composer.config_validation.utils import read_config_file
from mephisto.generators.form_composer.remote_procedures import JS_NAME_FUNCTION_MAPPING
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-@task_script(default_config_file="dynamic_presigned_urls_example_ec2_prolific")
-def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
-
- # Configure shared state
- task_data_config_path = os.path.join(
- os.path.dirname(os.path.abspath(__file__)),
- "data",
- "dynamic_presigned_urls",
- "task_data.json",
- )
- task_data = read_config_file(task_data_config_path)
- shared_state = SharedRemoteProcedureTaskState(
- static_task_data=task_data,
- function_registry=JS_NAME_FUNCTION_MAPPING,
- )
-
- operator.launch_task_run(cfg.mephisto, shared_state)
- operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-
-
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- build_command="build:presigned_urls",
- )
-
-
-def generate_data_json_config():
+def _generate_data_json_config():
"""
Generate extrapolated `task_data.json` config file,
based on existing form and tokens values config files
@@ -117,6 +54,30 @@ def generate_data_json_config():
)
+@task_script(default_config_file="dynamic_presigned_urls_example_ec2_prolific")
+def main(operator: Operator, cfg: DictConfig) -> None:
+ examples.build_form_composer_dynamic_presigned_urls_ec2_prolific(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
+
+ # Configure shared state
+ task_data_config_path = os.path.join(
+ os.path.dirname(os.path.abspath(__file__)),
+ "data",
+ "dynamic_presigned_urls",
+ "task_data.json",
+ )
+ task_data = read_config_file(task_data_config_path)
+ shared_state = SharedRemoteProcedureTaskState(
+ static_task_data=task_data,
+ function_registry=JS_NAME_FUNCTION_MAPPING,
+ )
+
+ operator.launch_task_run(cfg.mephisto, shared_state)
+ operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
+
+
if __name__ == "__main__":
- generate_data_json_config()
+ _generate_data_json_config()
main()
diff --git a/examples/form_composer_demo/run_task_with_gold_unit.py b/examples/form_composer_demo/run_task_with_gold_unit.py
index 603466b53..e02cffc24 100644
--- a/examples/form_composer_demo/run_task_with_gold_unit.py
+++ b/examples/form_composer_demo/run_task_with_gold_unit.py
@@ -22,51 +22,10 @@
from mephisto.abstractions.blueprints.mixins.use_gold_unit import get_gold_factory
from mephisto.abstractions.blueprints.mixins.use_gold_unit import UseGoldUnit
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:simple:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- build_command="dev:simple",
- )
-
-
def _get_gold_data() -> List[Dict[str, Any]]:
gold_data_path = os.path.join(
# Root project directory
@@ -90,7 +49,10 @@ def _get_gold_data() -> List[Dict[str, Any]]:
@task_script(default_config_file="example_local_mock_with_gold_unit")
def main(operator: Operator, cfg: DictConfig) -> None:
# 1. Build packages
- _build_custom_bundles(cfg)
+ examples.build_form_composer_simple_with_gold_unit(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
# 2. Prepare ShareState with Gold Units
gold_data = _get_gold_data()
diff --git a/examples/form_composer_demo/run_task_with_onboarding.py b/examples/form_composer_demo/run_task_with_onboarding.py
index a83c54a79..0b3167518 100644
--- a/examples/form_composer_demo/run_task_with_onboarding.py
+++ b/examples/form_composer_demo/run_task_with_onboarding.py
@@ -4,61 +4,18 @@
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.
-import os
-
from omegaconf import DictConfig
from mephisto.abstractions.blueprints.abstract.static_task.static_blueprint import (
SharedStaticTaskState,
)
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:simple:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- build_command="dev:simple",
- )
-
-
-def handle_onboarding(onboarding_data: dict) -> bool:
- if onboarding_data["outputs"]["success"] == True:
+def _handle_onboarding(onboarding_data: dict) -> bool:
+ if onboarding_data["outputs"]["success"] is True:
return True
return False
@@ -67,11 +24,14 @@ def handle_onboarding(onboarding_data: dict) -> bool:
@task_script(default_config_file="example_local_mock_with_oboarding")
def main(operator: Operator, cfg: DictConfig) -> None:
# 1. Build packages
- _build_custom_bundles(cfg)
+ examples.build_form_composer_simple_with_onboarding(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
# 2. Prepare ShareState with Onboarding
shared_state = SharedStaticTaskState(
- validate_onboarding=handle_onboarding,
+ validate_onboarding=_handle_onboarding,
)
# 3. Launch TaskRun
diff --git a/examples/form_composer_demo/run_task_with_screening.py b/examples/form_composer_demo/run_task_with_screening.py
index 531ec2c24..f8d3f1326 100644
--- a/examples/form_composer_demo/run_task_with_screening.py
+++ b/examples/form_composer_demo/run_task_with_screening.py
@@ -17,52 +17,11 @@
)
from mephisto.abstractions.blueprints.mixins.screen_task_required import ScreenTaskRequired
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:simple:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- build_command="dev:simple",
- )
-
-
-def screening_unit_factory() -> dict:
+def _screening_unit_factory() -> dict:
while True:
screening_data_path = os.path.join(
# Root project directory
@@ -90,7 +49,10 @@ def screening_unit_factory() -> dict:
@task_script(default_config_file="example_local_mock_with_screening")
def main(operator: Operator, cfg: DictConfig) -> None:
# 1. Build packages
- _build_custom_bundles(cfg)
+ examples.build_form_composer_simple_with_screening(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
# 2. Prepare ShareState with Screeining
shared_state = SharedStaticTaskState()
@@ -99,7 +61,7 @@ def main(operator: Operator, cfg: DictConfig) -> None:
cfg.mephisto,
validate_screening_unit,
)
- shared_state.screening_data_factory = screening_unit_factory()
+ shared_state.screening_data_factory = _screening_unit_factory()
shared_state.qualifications += ScreenTaskRequired.get_mixin_qualifications(
cfg.mephisto,
shared_state,
diff --git a/examples/form_composer_demo/run_task_with_worker_opinion.py b/examples/form_composer_demo/run_task_with_worker_opinion.py
index 0ecf806f9..e0e33344c 100644
--- a/examples/form_composer_demo/run_task_with_worker_opinion.py
+++ b/examples/form_composer_demo/run_task_with_worker_opinion.py
@@ -9,68 +9,19 @@
from omegaconf import DictConfig
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
@task_script(default_config_file="example_local_mock")
def main(operator: Operator, cfg: DictConfig) -> None:
os.environ["REACT_APP__WITH_WORKER_OPINION"] = "true"
-
- # Build packages
- _build_custom_bundles(cfg)
-
- operator.launch_task_run(cfg.mephisto)
- operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
-
-
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- mephisto_packages_dir = os.path.join(
- # Root project directory
- os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
- "packages",
- )
-
- # Build `mephisto-task-multipart` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-multipart",
- build_command="build",
- )
-
- # Build `mephisto-task-addons` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="mephisto-task-addons",
- build_command="build",
- )
-
- # Build `react-form-composer` React package
- build_custom_bundle(
- mephisto_packages_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="react-form-composer",
- build_command="build",
- )
-
- # Build Review UI for the application
- build_custom_bundle(
- cfg.task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- webapp_name="webapp",
- build_command="build:simple:review",
- )
-
- # Build Task UI for the application
- build_custom_bundle(
- cfg.task_dir,
+ examples.build_form_composer_simple_with_worker_opinion(
force_rebuild=cfg.mephisto.task.force_rebuild,
post_install_script=cfg.mephisto.task.post_install_script,
- build_command="dev:simple",
)
+ operator.launch_task_run(cfg.mephisto)
+ operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
if __name__ == "__main__":
diff --git a/examples/parlai_chat_task_demo/run_task.py b/examples/parlai_chat_task_demo/run_task.py
index fce6a02de..5013b13da 100644
--- a/examples/parlai_chat_task_demo/run_task.py
+++ b/examples/parlai_chat_task_demo/run_task.py
@@ -4,17 +4,19 @@
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.
-
import os
-from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import task_script
-from mephisto.operations.hydra_config import build_default_task_config
+from dataclasses import dataclass
+from dataclasses import field
+
+from omegaconf import DictConfig
+
from mephisto.abstractions.blueprints.parlai_chat.parlai_chat_blueprint import (
SharedParlAITaskState,
)
-
-from omegaconf import DictConfig
-from dataclasses import dataclass, field
+from mephisto.operations.hydra_config import build_default_task_config
+from mephisto.operations.operator import Operator
+from mephisto.tools.building_react_apps import examples
+from mephisto.tools.scripts import task_script
@dataclass
@@ -33,6 +35,10 @@ class ParlAITaskConfig(build_default_task_config("example")): # type: ignore
@task_script(config=ParlAITaskConfig)
def main(operator: "Operator", cfg: DictConfig) -> None:
+ examples.build_parlai_chat_task_demo(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
world_opt = {"num_turns": cfg.num_turns, "turn_timeout": cfg.turn_timeout}
diff --git a/examples/remote_procedure/mnist/model.py b/examples/remote_procedure/mnist/model.py
index 7e2299cb2..57fc46699 100644
--- a/examples/remote_procedure/mnist/model.py
+++ b/examples/remote_procedure/mnist/model.py
@@ -7,17 +7,22 @@
# Using the implementation provided at
# https://github.com/aaron-xichen/pytorch-playground/blob/master/mnist/model.py
-import torch.nn as nn
from collections import OrderedDict
+
+import torch.nn as nn
import torch.utils.model_zoo as model_zoo
-model_urls = {"mnist": "http://ml.cs.tsinghua.edu.cn/~chenxi/pytorch-models/mnist-b07bb66b.pth"}
+model_urls = {
+ "mnist": "http://ml.cs.tsinghua.edu.cn/~chenxi/pytorch-models/mnist-b07bb66b.pth",
+}
class MLP(nn.Module):
def __init__(self, input_dims, n_hiddens, n_class):
super(MLP, self).__init__()
+
assert isinstance(input_dims, int), "Please provide int for input_dims"
+
self.input_dims = input_dims
current_dims = input_dims
layers = OrderedDict()
@@ -26,31 +31,38 @@ def __init__(self, input_dims, n_hiddens, n_class):
n_hiddens = [n_hiddens]
else:
n_hiddens = list(n_hiddens)
+
for i, n_hidden in enumerate(n_hiddens):
layers["fc{}".format(i + 1)] = nn.Linear(current_dims, n_hidden)
layers["relu{}".format(i + 1)] = nn.ReLU()
layers["drop{}".format(i + 1)] = nn.Dropout(0.2)
current_dims = n_hidden
+
layers["out"] = nn.Linear(current_dims, n_class)
self.model = nn.Sequential(layers)
print(self.model)
- def forward(self, input):
+ def forward(self, _input):
# input = input.view(input.size(0), -1)
- assert input.size(1) == self.input_dims
- return self.model.forward(input)
+ assert _input.size(1) == self.input_dims
+
+ return self.model.forward(_input)
-def mnist(input_dims=784, n_hiddens=[256, 256], n_class=10, pretrained=None):
+def mnist(input_dims=784, n_hiddens=[256, 256], n_class=10, pretrained=None) -> MLP:
model = MLP(input_dims, n_hiddens, n_class)
+
if pretrained is not None:
try:
m = model_zoo.load_url(model_urls["mnist"])
except RuntimeError:
# No GPU, CPU only
m = model_zoo.load_url(model_urls["mnist"], map_location="cpu")
+
state_dict = m.state_dict() if isinstance(m, nn.Module) else m
assert isinstance(state_dict, (dict, OrderedDict)), type(state_dict)
+
model.load_state_dict(state_dict)
+
return model
diff --git a/examples/remote_procedure/mnist/run_task.py b/examples/remote_procedure/mnist/run_task.py
index 8446aee38..ac72b9cd4 100644
--- a/examples/remote_procedure/mnist/run_task.py
+++ b/examples/remote_procedure/mnist/run_task.py
@@ -9,34 +9,39 @@
from PIL import Image
except ImportError:
print(
- "Need to have torch, PIL, numpy installed to use this demo. For example: pip install torch pillow numpy"
+ "Need to have torch, PIL, numpy installed to use this demo. "
+ "For example: pip install torch pillow numpy"
)
exit(1)
import base64
from io import BytesIO
+from typing import Any
+from typing import Dict
+from typing import List
+
+from omegaconf import DictConfig
+
+from examples.remote_procedure.mnist.model import mnist
from mephisto.abstractions.blueprints.mixins.screen_task_required import (
ScreenTaskRequired,
)
-from mephisto.data_model.unit import Unit
-from model import mnist
-
-from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import (
- build_custom_bundle,
- task_script,
+from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
+ RemoteProcedureAgentState,
)
from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
SharedRemoteProcedureTaskState,
- RemoteProcedureAgentState,
)
+from mephisto.data_model.unit import Unit
+from mephisto.operations.operator import Operator
+from mephisto.tools.building_react_apps import examples
+from mephisto.tools.scripts import task_script
+from mephisto.utils.console_writer import ConsoleWriter
-from omegaconf import DictConfig
-from typing import List, Any, Dict
-from rich import print
+logger = ConsoleWriter()
-def my_screening_unit_generator():
+def _my_screening_unit_generator() -> dict:
"""
The frontend react webapp reads in
isScreeningUnit using the initialTaskData
@@ -48,7 +53,7 @@ def my_screening_unit_generator():
}
-def validate_screening_unit(unit: Unit):
+def _validate_screening_unit(unit: Unit) -> bool:
"""Checking if the drawn number is 3"""
agent = unit.get_assigned_agent()
if agent is not None:
@@ -60,32 +65,45 @@ def validate_screening_unit(unit: Unit):
return False
+def _handle_with_model(
+ _request_id: str,
+ args: Dict[str, Any],
+ agent_state: RemoteProcedureAgentState,
+) -> Dict[str, Any]:
+ """Convert the image to be read by MNIST classifier, then classify"""
+
+ img_dat = args["urlData"].split("data:image/png;base64,")[1]
+ im = Image.open(BytesIO(base64.b64decode(img_dat)))
+ im_gray = im.convert("L")
+ im_resized = im_gray.resize((28, 28))
+ im_vals = list(im_resized.getdata())
+ norm_vals = [(255 - x) * 1.0 / 255.0 for x in im_vals]
+ in_tensor = torch.tensor([norm_vals])
+
+ mnist_model = mnist(pretrained=True)
+ output = mnist_model(in_tensor)
+
+ pred = output.data.max(1)[1]
+
+ logger.debug(f"Predicted digit: {pred.item()}")
+
+ return {
+ "digit_prediction": pred.item(),
+ }
+
+
@task_script(default_config_file="example_local_mock")
def main(operator: Operator, cfg: DictConfig) -> None:
+ examples.build_remote_procedure_mnist(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
+
tasks: List[Dict[str, Any]] = [{"isScreeningUnit": False}] * cfg.num_tasks
- mnist_model = mnist(pretrained=True)
is_using_screening_units = cfg.mephisto.blueprint["use_screening_task"]
- def handle_with_model(
- _request_id: str, args: Dict[str, Any], agent_state: RemoteProcedureAgentState
- ) -> Dict[str, Any]:
- """Convert the image to be read by MNIST classifier, then classify"""
- img_dat = args["urlData"].split("data:image/png;base64,")[1]
- im = Image.open(BytesIO(base64.b64decode(img_dat)))
- im_gray = im.convert("L")
- im_resized = im_gray.resize((28, 28))
- im_vals = list(im_resized.getdata())
- norm_vals = [(255 - x) * 1.0 / 255.0 for x in im_vals]
- in_tensor = torch.tensor([norm_vals])
- output = mnist_model(in_tensor)
- pred = output.data.max(1)[1]
- print("Predicted digit:", pred.item())
- return {
- "digit_prediction": pred.item(),
- }
-
function_registry = {
- "classify_digit": handle_with_model,
+ "classify_digit": _handle_with_model,
}
shared_state = SharedRemoteProcedureTaskState(
static_task_data=tasks,
@@ -96,21 +114,13 @@ def handle_with_model(
# You have to define a few more properties to enable screening units
shared_state.on_unit_submitted = ScreenTaskRequired.create_validation_function(
cfg.mephisto,
- validate_screening_unit,
+ _validate_screening_unit,
)
- shared_state.screening_data_factory = my_screening_unit_generator()
+ shared_state.screening_data_factory = _my_screening_unit_generator()
shared_state.qualifications += ScreenTaskRequired.get_mixin_qualifications(
cfg.mephisto, shared_state
)
- task_dir = cfg.task_dir
-
- build_custom_bundle(
- task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
-
operator.launch_task_run(cfg.mephisto, shared_state)
operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
diff --git a/examples/remote_procedure/mnist/webapp/package.json b/examples/remote_procedure/mnist/webapp/package.json
index 05c0debca..ba91f7c06 100644
--- a/examples/remote_procedure/mnist/webapp/package.json
+++ b/examples/remote_procedure/mnist/webapp/package.json
@@ -1,5 +1,5 @@
{
- "name": "parlai-mturk-task-compiler",
+ "name": "remote-procedure-mnist-example",
"version": "1.0.1",
"description": "",
"main": "webpack.config.js",
diff --git a/examples/remote_procedure/mnist/webapp/src/app.jsx b/examples/remote_procedure/mnist/webapp/src/app.jsx
index 4ec316575..729608c76 100644
--- a/examples/remote_procedure/mnist/webapp/src/app.jsx
+++ b/examples/remote_procedure/mnist/webapp/src/app.jsx
@@ -16,7 +16,7 @@ import {
MephistoContext,
useMephistoRemoteProcedureTask,
ErrorBoundary,
-} from "mephisto-task";
+} from "mephisto-task-multipart";
/* ================= Application Components ================= */
diff --git a/examples/remote_procedure/mnist/webapp/webpack.config.js b/examples/remote_procedure/mnist/webapp/webpack.config.js
index 7c92f67df..5438d5a60 100644
--- a/examples/remote_procedure/mnist/webapp/webpack.config.js
+++ b/examples/remote_procedure/mnist/webapp/webpack.config.js
@@ -16,6 +16,11 @@ module.exports = {
resolve: {
alias: {
react: path.resolve("./node_modules/react"),
+ // Use local library with code that can submit FormData
+ "mephisto-task-multipart": path.resolve(
+ __dirname,
+ "../../../../packages/mephisto-task-multipart"
+ ),
},
fallback: {
net: false,
diff --git a/examples/remote_procedure/mnist/webapp/webpack.config.review.js b/examples/remote_procedure/mnist/webapp/webpack.config.review.js
index 5e226ee6c..68bd1bc4f 100644
--- a/examples/remote_procedure/mnist/webapp/webpack.config.review.js
+++ b/examples/remote_procedure/mnist/webapp/webpack.config.review.js
@@ -16,6 +16,11 @@ module.exports = {
resolve: {
alias: {
react: path.resolve("./node_modules/react"),
+ // Use local library with code that can submit FormData
+ "mephisto-task-multipart": path.resolve(
+ __dirname,
+ "../../../../packages/mephisto-task-multipart"
+ ),
},
fallback: {
net: false,
diff --git a/examples/remote_procedure/template/run_task.py b/examples/remote_procedure/template/run_task.py
index ef4642f42..e1b54f4c2 100644
--- a/examples/remote_procedure/template/run_task.py
+++ b/examples/remote_procedure/template/run_task.py
@@ -4,21 +4,27 @@
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.
-from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import (
- task_script,
- build_custom_bundle,
+from typing import Any
+from typing import Dict
+from typing import List
+
+from omegaconf import DictConfig
+
+from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
+ RemoteProcedureAgentState,
)
from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
SharedRemoteProcedureTaskState,
- RemoteProcedureAgentState,
)
+from mephisto.operations.operator import Operator
+from mephisto.tools.building_react_apps import examples
+from mephisto.tools.scripts import task_script
+from mephisto.utils.console_writer import ConsoleWriter
-from omegaconf import DictConfig
-from typing import Any, Dict
+logger = ConsoleWriter()
-def build_local_context(num_tasks):
+def _build_local_context(num_tasks: int) -> dict:
"""
Create local context that you don't intend to be shared with the frontend,
but which you may want your remote functions to use
@@ -28,10 +34,11 @@ def build_local_context(num_tasks):
context = {}
for x in range(num_tasks):
context[x] = f"Hello {x}, this task was for you."
+
return context
-def build_tasks(num_tasks):
+def _build_tasks(num_tasks: int) -> List[dict]:
"""
Create a set of tasks you want annotated
"""
@@ -44,52 +51,59 @@ def build_tasks(num_tasks):
"local_value_key": x,
}
)
+
return tasks
+def _onboarding_always_valid(onboarding_data: dict) -> bool:
+ # NOTE you can make an onboarding task and validate it here
+ logger.debug(f"Onboarding data: {onboarding_data}")
+ return True
+
+
@task_script(default_config_file="example_local_mock")
def main(operator: Operator, cfg: DictConfig) -> None:
- def onboarding_always_valid(onboarding_data):
- # NOTE you can make an onboarding task and validate it here
- print(onboarding_data)
- return True
+ examples.build_remote_procedure_template(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
# Right now we're building locally, but should eventually
# use non-local for the real thing
- tasks = build_tasks(cfg.num_tasks)
- context = build_local_context(cfg.num_tasks)
+ tasks = _build_tasks(cfg.num_tasks)
+ context = _build_local_context(cfg.num_tasks)
- def handle_with_model(
- _request_id: str, args: Dict[str, Any], agent_state: RemoteProcedureAgentState
+ def _handle_with_model(
+ _request_id: str,
+ args: Dict[str, Any],
+ agent_state: RemoteProcedureAgentState,
) -> Dict[str, Any]:
"""Remote call to process external content using a 'model'"""
# NOTE this body can be whatever you want
- print(f"The parsed args are {args}, you can do what you want with that")
- print(f"You can also use {agent_state.init_data}, to get task keys")
+ logger.debug(f"The parsed args are {args}, you can do what you want with that")
+ logger.debug(f"You can also use {agent_state.init_data}, to get task keys")
+
assert agent_state.init_data is not None
+
idx = agent_state.init_data["local_value_key"]
- print(f"And that may let you get local context, like {context[idx]}")
+
+ logger.debug(f"And that may let you get local context, like {context[idx]}")
+
return {
"secret_local_value": context[idx],
"update": f"this was request {args['arg3'] + 1}",
}
function_registry = {
- "handle_with_model": handle_with_model,
+ "handle_with_model": _handle_with_model,
}
shared_state = SharedRemoteProcedureTaskState(
static_task_data=tasks,
- validate_onboarding=onboarding_always_valid,
+ validate_onboarding=_onboarding_always_valid,
function_registry=function_registry,
)
- task_dir = cfg.task_dir
- build_custom_bundle(
- task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
operator.launch_task_run(cfg.mephisto, shared_state)
operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
diff --git a/examples/remote_procedure/template/webapp/package.json b/examples/remote_procedure/template/webapp/package.json
index 9e1055d88..df44ad125 100644
--- a/examples/remote_procedure/template/webapp/package.json
+++ b/examples/remote_procedure/template/webapp/package.json
@@ -1,5 +1,5 @@
{
- "name": "parlai-mturk-task-compiler",
+ "name": "remote-procedure-template-example",
"version": "1.0.1",
"description": "",
"main": "webpack.config.js",
diff --git a/examples/remote_procedure/template/webapp/src/app.jsx b/examples/remote_procedure/template/webapp/src/app.jsx
index d05ed8427..edc28c90c 100644
--- a/examples/remote_procedure/template/webapp/src/app.jsx
+++ b/examples/remote_procedure/template/webapp/src/app.jsx
@@ -12,7 +12,7 @@ import {
MephistoContext,
useMephistoRemoteProcedureTask,
ErrorBoundary,
-} from "mephisto-task";
+} from "mephisto-task-multipart";
/* ================= Application Components ================= */
diff --git a/examples/remote_procedure/template/webapp/webpack.config.js b/examples/remote_procedure/template/webapp/webpack.config.js
index 7c92f67df..5438d5a60 100644
--- a/examples/remote_procedure/template/webapp/webpack.config.js
+++ b/examples/remote_procedure/template/webapp/webpack.config.js
@@ -16,6 +16,11 @@ module.exports = {
resolve: {
alias: {
react: path.resolve("./node_modules/react"),
+ // Use local library with code that can submit FormData
+ "mephisto-task-multipart": path.resolve(
+ __dirname,
+ "../../../../packages/mephisto-task-multipart"
+ ),
},
fallback: {
net: false,
diff --git a/examples/remote_procedure/toxicity_detection/run_task.py b/examples/remote_procedure/toxicity_detection/run_task.py
index 681e10576..6deb6586a 100644
--- a/examples/remote_procedure/toxicity_detection/run_task.py
+++ b/examples/remote_procedure/toxicity_detection/run_task.py
@@ -10,24 +10,25 @@
print("Need to have detoxify to use this demo. For example: pip install detoxify")
exit(1)
-from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import (
- build_custom_bundle,
- task_script,
-)
+from typing import Any
+from typing import Dict
+from typing import List
+
+from omegaconf import DictConfig
from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
- SharedRemoteProcedureTaskState,
RemoteProcedureAgentState,
)
-from omegaconf import DictConfig
-from typing import Any, Dict
+from mephisto.abstractions.blueprints.remote_procedure.remote_procedure_blueprint import (
+ SharedRemoteProcedureTaskState,
+)
+from mephisto.operations.operator import Operator
+from mephisto.tools.building_react_apps import examples
+from mephisto.tools.scripts import task_script
-def build_tasks(num_tasks):
- """
- Create a set of tasks you want annotated
- """
+def _build_tasks(num_tasks: int) -> List[dict]:
+ """Create a set of tasks you want annotated"""
# NOTE These form the init_data for a task
tasks = []
for x in range(num_tasks):
@@ -37,26 +38,35 @@ def build_tasks(num_tasks):
"local_value_key": x,
}
)
+
return tasks
-def determine_toxicity(text: str):
+def _determine_toxicity(text: str) -> str:
return Detoxify("original").predict(text)["toxicity"]
+def _calculate_toxicity(
+ _request_id: str,
+ args: Dict[str, Any],
+ agent_state: RemoteProcedureAgentState,
+) -> Dict[str, Any]:
+ return {
+ "toxicity": str(_determine_toxicity(args["text"])),
+ }
+
+
@task_script(default_config_file="example_local_mock")
def main(operator: Operator, cfg: DictConfig) -> None:
- tasks = build_tasks(cfg.num_tasks)
+ examples.build_remote_procedure_toxicity_detection(
+ force_rebuild=cfg.mephisto.task.force_rebuild,
+ post_install_script=cfg.mephisto.task.post_install_script,
+ )
- def calculate_toxicity(
- _request_id: str, args: Dict[str, Any], agent_state: RemoteProcedureAgentState
- ) -> Dict[str, Any]:
- return {
- "toxicity": str(determine_toxicity(args["text"])),
- }
+ tasks = _build_tasks(cfg.num_tasks)
function_registry = {
- "determine_toxicity": calculate_toxicity,
+ "determine_toxicity": _calculate_toxicity,
}
shared_state = SharedRemoteProcedureTaskState(
@@ -64,13 +74,6 @@ def calculate_toxicity(
function_registry=function_registry,
)
- task_dir = cfg.task_dir
- build_custom_bundle(
- task_dir,
- force_rebuild=cfg.mephisto.task.force_rebuild,
- post_install_script=cfg.mephisto.task.post_install_script,
- )
-
operator.launch_task_run(cfg.mephisto, shared_state)
operator.wait_for_runs_then_shutdown(skip_input=True, log_rate=30)
diff --git a/examples/remote_procedure/toxicity_detection/webapp/package.json b/examples/remote_procedure/toxicity_detection/webapp/package.json
index 90dbf718a..24259b2e8 100644
--- a/examples/remote_procedure/toxicity_detection/webapp/package.json
+++ b/examples/remote_procedure/toxicity_detection/webapp/package.json
@@ -1,5 +1,5 @@
{
- "name": "parlai-mturk-task-compiler",
+ "name": "remote-procedure-toxicity-detection-example",
"version": "1.0.1",
"description": "",
"main": "webpack.config.js",
diff --git a/examples/remote_procedure/toxicity_detection/webapp/src/app.jsx b/examples/remote_procedure/toxicity_detection/webapp/src/app.jsx
index 8fde25f54..50111c8ac 100644
--- a/examples/remote_procedure/toxicity_detection/webapp/src/app.jsx
+++ b/examples/remote_procedure/toxicity_detection/webapp/src/app.jsx
@@ -16,7 +16,7 @@ import {
MephistoContext,
useMephistoRemoteProcedureTask,
ErrorBoundary,
-} from "mephisto-task";
+} from "mephisto-task-multipart";
function RemoteProcedureApp() {
let mephistoProps = useMephistoRemoteProcedureTask({});
diff --git a/examples/remote_procedure/toxicity_detection/webapp/webpack.config.js b/examples/remote_procedure/toxicity_detection/webapp/webpack.config.js
index 7c92f67df..5438d5a60 100644
--- a/examples/remote_procedure/toxicity_detection/webapp/webpack.config.js
+++ b/examples/remote_procedure/toxicity_detection/webapp/webpack.config.js
@@ -16,6 +16,11 @@ module.exports = {
resolve: {
alias: {
react: path.resolve("./node_modules/react"),
+ // Use local library with code that can submit FormData
+ "mephisto-task-multipart": path.resolve(
+ __dirname,
+ "../../../../packages/mephisto-task-multipart"
+ ),
},
fallback: {
net: false,
diff --git a/examples/static_react_task/run_task.py b/examples/static_react_task/run_task.py
index 643efed65..5f438a699 100644
--- a/examples/static_react_task/run_task.py
+++ b/examples/static_react_task/run_task.py
@@ -15,11 +15,11 @@
)
from mephisto.data_model.unit import Unit
from mephisto.operations.operator import Operator
-from mephisto.tools.scripts import build_custom_bundle
+from mephisto.tools.building_react_apps import examples
from mephisto.tools.scripts import task_script
-def my_screening_unit_generator():
+def _my_screening_unit_generator() -> dict:
while True:
yield {
"text": "SCREENING UNIT: Press the red button",
@@ -27,11 +27,12 @@ def my_screening_unit_generator():
}
-def validate_screening_unit(unit: Unit):
+def _validate_screening_unit(unit: Unit) -> bool:
agent = unit.get_assigned_agent()
if agent is not None:
data = agent.state.get_data()
print(data)
+
if (
data["outputs"] is not None
and "rating" in data["outputs"]
@@ -39,36 +40,31 @@ def validate_screening_unit(unit: Unit):
):
# User pressed the red button
return True
+
return False
-def handle_onboarding(onboarding_data):
- if onboarding_data["outputs"]["success"] == True:
+def _handle_onboarding(onboarding_data: dict) -> bool:
+ if onboarding_data["outputs"]["success"] is True:
return True
+
return False
-def _build_custom_bundles(cfg: DictConfig) -> None:
- """Locally build bundles that are not available on npm repository"""
- build_custom_bundle(
- cfg.task_dir,
+@task_script(default_config_file="example_local_mock.yaml")
+def main(operator: Operator, cfg: DictConfig) -> None:
+ examples.build_static_react_task(
force_rebuild=cfg.mephisto.task.force_rebuild,
post_install_script=cfg.mephisto.task.post_install_script,
)
-
-@task_script(default_config_file="example_local_mock.yaml")
-def main(operator: Operator, cfg: DictConfig) -> None:
- # Build packages
- _build_custom_bundles(cfg)
-
is_using_screening_units = cfg.mephisto.blueprint["use_screening_task"]
shared_state = SharedStaticTaskState(
static_task_data=[
{"text": "This text is good text!"},
{"text": "This text is bad text!"},
],
- validate_onboarding=handle_onboarding,
+ validate_onboarding=_handle_onboarding,
)
if is_using_screening_units:
@@ -76,9 +72,9 @@ def main(operator: Operator, cfg: DictConfig) -> None:
# few more properties set on shared_state
shared_state.on_unit_submitted = ScreenTaskRequired.create_validation_function(
cfg.mephisto,
- validate_screening_unit,
+ _validate_screening_unit,
)
- shared_state.screening_data_factory = my_screening_unit_generator()
+ shared_state.screening_data_factory = _my_screening_unit_generator()
shared_state.qualifications += ScreenTaskRequired.get_mixin_qualifications(
cfg.mephisto, shared_state
)
diff --git a/examples/static_react_task/webapp/cypress/e2e/static_react_task.cy.js b/examples/static_react_task/webapp/cypress/e2e/static_react_task.cy.js
index e581e517b..beda3c177 100644
--- a/examples/static_react_task/webapp/cypress/e2e/static_react_task.cy.js
+++ b/examples/static_react_task/webapp/cypress/e2e/static_react_task.cy.js
@@ -12,6 +12,7 @@ describe("Loads static_react_task", () => {
expect(interception.response.statusCode).to.eq(200);
});
});
+
it("Loads correct react elements", () => {
cy.get('[data-cy="directions-container"]');
cy.get('[data-cy="task-data-text"]');
@@ -22,13 +23,16 @@ describe("Loads static_react_task", () => {
describe("Submits static_react_task", () => {
it("Gets request from pressing good button", () => {
+ cy.reload();
cy.intercept({ pathname: "/submit_task" }).as("goodTaskSubmit");
cy.get('[data-cy="good-button"]').click();
cy.wait("@goodTaskSubmit").then((interception) => {
expect(interception.response.statusCode).to.eq(200);
});
});
+
it("Shows alert from pressing good button", () => {
+ cy.reload();
cy.on("window:alert", (txt) => {
expect(txt).to.contains(
'The task has been submitted! Data: {"rating":"good"}'
@@ -38,6 +42,7 @@ describe("Submits static_react_task", () => {
});
it("Gets request from pressing bad button", () => {
+ cy.reload();
cy.intercept({ pathname: "/submit_task" }).as("badTaskSubmit");
cy.get('[data-cy="bad-button"]').click();
cy.wait("@badTaskSubmit").then((interception) => {
@@ -46,6 +51,7 @@ describe("Submits static_react_task", () => {
});
it("Shows alert from pressing bad button", () => {
+ cy.reload();
cy.on("window:alert", (txt) => {
expect(txt).to.contains(
'The task has been submitted! Data: {"rating":"bad"}'
diff --git a/examples/static_react_task/webapp/src/components/core_components.jsx b/examples/static_react_task/webapp/src/components/core_components.jsx
index 5a7a59e23..7a7b65af7 100644
--- a/examples/static_react_task/webapp/src/components/core_components.jsx
+++ b/examples/static_react_task/webapp/src/components/core_components.jsx
@@ -44,7 +44,7 @@ function LoadingScreen() {
function Directions({ children }) {
return (
-
+
);
@@ -61,12 +61,15 @@ function SimpleFrontend({ taskData, isOnboarding, onSubmit, onError }) {
-
{taskData.text}
+
+ {taskData.text}
+
{!resonseSubmitted && (