diff --git a/.github/workflows/automerge_plugin-only_prs.yml b/.github/workflows/automerge_plugin-only_prs.yml index 1c3cef18..1897be11 100644 --- a/.github/workflows/automerge_plugin-only_prs.yml +++ b/.github/workflows/automerge_plugin-only_prs.yml @@ -40,8 +40,9 @@ jobs: echo "::set-output name=BSC_DATABASESECRET:: ${{ secrets.BSC_DATABASESECRET_PROD }}" fi + isautomerge: - name: Check if PR is labeled with 'automerge' or 'automerge-web' + name: Set as 'automerge' if PR is labeled with 'automerge' or 'automerge-web' runs-on: ubuntu-latest if: | contains( github.event.pull_request.labels.*.name, 'automerge') || @@ -54,6 +55,7 @@ jobs: run: | echo "::set-output name=AUTOMERGE::True" + travis_success: name: Check if Travis build is successful runs-on: ubuntu-latest @@ -66,7 +68,7 @@ jobs: id: gettravisstatus run: | echo ${{ github.event.pull_request.head.sha }} - echo "TRAVIS_CONCLUSION=$(curl -X GET https://api.github.com/repos/brain-score/language/commits/${{ github.event.pull_request.head.sha }}/check-runs | python -c "from __future__ import print_function; import sys,json; print(next(run['conclusion'] for run in json.load(sys.stdin)['check_runs'] if run['name'] == 'Travis CI - Pull Request'))")" >> $GITHUB_ENV + echo "TRAVIS_CONCLUSION=$(python -c "import requests; r = requests.get(\"https://api.github.com/repos/brain-score/language/commits/$github.event.pull_request.head.sha/check-runs\"); print(next(run['conclusion'] for run in r.json()['check_runs'] if run['name'] == 'Travis CI - Pull Request'))")" >> $GITHUB_ENV - name: Check if Travis was successful id: istravisok run: | @@ -81,89 +83,35 @@ jobs: fi echo "::set-output name=TRAVIS_OK::$travisok" - email_on_failure: - name: If tests fail on an automerge-web PR, send email to submitter - runs-on: ubuntu-latest - needs: [setup, travis_success] - env: - BSC_DATABASESECRET: ${{ needs.setup.outputs.BSC_DATABASESECRET }} - if: | - needs.travis_success.outputs.TRAVIS_OK == 'False' && - join(github.event.pull_request.labels.*.name, ',') == 'automerge-web' - steps: - - name: Parse Brain-Score user ID from PR title - run: | - echo "BS_UID="$(<<<${{ github.event.pull_request.title }} | sed -E 's/.*\(user:([^)]+)\).*/\1/'"" >> $GITHUB_ENV - - name: Build project - run: | - python -m pip install "." - - name: Get email from Brain-Score UID - run: | - echo "TO_EMAIL=$(python .github/workflows/call_endpoints.py get_user_email '${{ env.BS_UID }}')" >> $GITHUB_ENV - - name: Send email to Brain-Score user - uses: dawidd6/action-send-mail@v3 - with: - # Required mail server address: - server_address: smtp.gmail.com - # Server port, default 25: - server_port: 465 - # Optional whether this connection use TLS (default is true if server_port is 465) - secure: true - # Optional (recommended) mail server username: - username: ${{ secrets.MAIL_USERNAME }} - # Optional (recommended) mail server password: - password: ${{ secrets.MAIL_PASSWORD }} - # Required mail subject: - subject: Brain-Score submission failed - # Required recipients' addresses: - to: ${{ env.BS_UID }} - # Required sender full name: - from: Brain-Score - # Optional plain body: - body: Your Brain-Score submission did not pass checks. Please review the test results and update the PR at https://github.com/brain-score/language/pull/${{ github.event.number }} or send in an updated submission via the website. - - checkchanges: - name: Check if PR only changes plugin files + + plugin_only: + name: Ensure PR ONLY changes plugin files runs-on: ubuntu-latest needs: travis_success if: ${{ needs.travis_success.outputs.TRAVIS_OK == 'True' }} outputs: - PLUGIN_INFO: ${{ steps.getpluginfo.outputs.PLUGIN_INFO }} - IS_PLUGIN_ONLY: ${{ steps.ispluginonly.outputs.IS_PLUGIN_ONLY }} + PLUGIN_ONLY: ${{ steps.ispluginonly.outputs.PLUGIN_ONLY }} steps: - - name: Check out repository code - uses: actions/checkout@v2 - - - name: Get changed files - uses: dorny/paths-filter@v2.11.1 - id: filter - with: - list-files: shell - filters: | - changed: - - '**' - - name: Save changed files to env var - run: echo "CHANGED_FILES=${{ steps.filter.outputs.changed_files }}" >> $GITHUB_ENV - - - name: Parse changed files with python script - run: | - echo "PLUGIN_INFO=$(python .github/workflows/parse_changed_files.py '${{ env.CHANGED_FILES }}')" >> $GITHUB_ENV - - - name: Save plugin info to outputs - id: getpluginfo - run: | - echo "PLUGIN_INFO=$PLUGIN_INFO" >> $GITHUB_OUTPUT - - - name: check if plugin-only + - name: Parse plugin_only confirmation from Travis status update + id: getpluginonlyvalue + run: echo "PLUGIN_ONLY=$(python -c "import requests; r = requests.get(\"https://api.github.com/repos/brain-score/language/statuses/$github.event.pull_request.head.sha\"); print(next(status['description'].split('- ')[1] for status in r.json() if status['description'].startswith('Run automerge workflow')))")" >> $GITHUB_ENV + - name: Check if PR is plugin only id: ispluginonly run: | - echo "IS_PLUGIN_ONLY=$(jq -r '.is_plugin_only' <<< "$PLUGIN_INFO")" >> $GITHUB_OUTPUT - + if [ "$PLUGIN_ONLY" == "True" ] + then + pluginonly=True + else + pluginonly=False + fi + echo "::set-output name=PLUGIN_ONLY::$pluginonly" + + automerge: name: If plugin-only, approve and merge runs-on: ubuntu-latest - needs: checkchanges - if: ${{ needs.checkchanges.outputs.IS_PLUGIN_ONLY == 'True' }} + needs: plugin_only + if: ${{ needs.plugin_only.outputs.PLUGIN_ONLY == 'True' }} steps: - name: Auto Approve uses: hmarr/auto-approve-action@v3.1.0 diff --git a/.github/workflows/call_endpoints.py b/.github/workflows/call_endpoints.py deleted file mode 100644 index 053971fe..00000000 --- a/.github/workflows/call_endpoints.py +++ /dev/null @@ -1,10 +0,0 @@ -import ast -import sys - -from brainscore_language.submission.endpoints import call_jenkins, get_email_from_uid - - -if __name__ == '__main__': - function = getattr(sys.modules[__name__], sys.argv[1]) - args = ast.literal_eval(sys.argv[2]) - function(args) \ No newline at end of file diff --git a/.github/workflows/parse_changed_files.py b/.github/workflows/parse_changed_files.py deleted file mode 100644 index 511b8ca7..00000000 --- a/.github/workflows/parse_changed_files.py +++ /dev/null @@ -1,94 +0,0 @@ -import re -import sys -from pathlib import Path -from typing import List, Tuple - -PLUGIN_DIRS = ['models', 'benchmarks', 'data', 'metrics'] - - -def get_changed_files(changed_files: str) -> Tuple[List[str], List[str]]: - changed_files_list = changed_files.split() - - plugin_files_changed = [] - non_plugin_files_changed = [] - - for f in changed_files_list: - if not any(plugin_dir in f for plugin_dir in PLUGIN_DIRS): - non_plugin_files_changed.append(f) - else: - plugin_files_changed.append(f) - - return plugin_files_changed, non_plugin_files_changed - - -def is_plugin_only(plugins_dict: dict, non_plugin_files_changed: List[str]): - """ - Stores `plugins_dict['is_plugin_only']` `"True"` or `"False"` - depending on whether there are any changed non-plugin files. - """ - plugin_only = len(non_plugin_files_changed) > 0 - plugins_dict["is_plugin_only"] = "False" if plugin_only else "True" - - -def _get_registered_plugins(plugin_type: str, plugin_dirs: List[str]) -> List[str]: - """ - Searches all `plugin_type` __init.py__ files for registered plugins. - Returns list of identifiers for each registered plugin. - """ - registered_plugins = [] - - plugin_type_dir = Path(f'brainscore_language/{plugin_type}') - - for plugin_dirname in plugin_dirs: - plugin_dirpath = plugin_type_dir / plugin_dirname - init_file = plugin_dirpath / "__init__.py" - with open(init_file) as f: - registry_name = plugin_type.strip( - 's') + '_registry' # remove plural and determine variable name, e.g. "models" -> "model_registry" - plugin_registrations = [line for line in f if f"{registry_name}[" - in line.replace('\"', '\'')] - for line in plugin_registrations: - result = re.search(f'{registry_name}\[.*\]', line) - identifier = result.group(0)[len(registry_name) + 2:-2] # remove brackets and quotes - registered_plugins.append(identifier) - - return registered_plugins - - -def plugins_to_score(plugins_dict, plugin_files_changed) -> str: - plugins_dict["run_score"] = "False" - - scoring_plugin_types = ("models", "benchmarks") - scoring_plugin_paths = tuple(f'brainscore_language/{plugin_type}/' for plugin_type in scoring_plugin_types) - model_and_benchmark_files = [fname for fname in plugin_files_changed if fname.startswith(scoring_plugin_paths)] - if len(model_and_benchmark_files) > 0: - plugins_dict["run_score"] = "True" - for plugin_type in scoring_plugin_types: - plugin_dirs = set( - [plugin_name_from_path(fname) for fname in model_and_benchmark_files if f'/{plugin_type}/' in fname]) - plugins_to_score = _get_registered_plugins(plugin_type, plugin_dirs) - plugins_dict[f'new_{plugin_type}'] = ' '.join(plugins_to_score) - - -def serialize_dict(plugins_dict: dict) -> str: - return str(plugins_dict).replace('\'', '\"') - - -def plugin_name_from_path(path_relative_to_library: str) -> str: - """ - Returns the name of the plugin from the given path. - E.g. `plugin_name_from_path("brainscore_vision/models/mymodel")` will return `"mymodel"`. - """ - return path_relative_to_library.split('/')[2] - - -if __name__ == '__main__': - changed_files = sys.argv[1] - plugin_files_changed, non_plugin_files_changed = get_changed_files(changed_files) - - plugins_dict = {} - is_plugin_only(plugins_dict, non_plugin_files_changed) - plugins_to_score(plugins_dict, plugin_files_changed) - plugins_dict = serialize_dict(plugins_dict) - - print(plugins_dict) diff --git a/.github/workflows/score_new_plugins.yml b/.github/workflows/score_new_plugins.yml index e17db32e..5fc5a178 100644 --- a/.github/workflows/score_new_plugins.yml +++ b/.github/workflows/score_new_plugins.yml @@ -35,6 +35,7 @@ jobs: echo "::set-output name=BSC_DATABASESECRET:: ${{ secrets.BSC_DATABASESECRET_PROD }}" fi + changes_models_or_benchmarks: name: Check if PR makes changes to /models or /benchmarks runs-on: ubuntu-latest @@ -44,32 +45,19 @@ jobs: steps: - name: Check out repository code uses: actions/checkout@v2 - - - name: Get changed files - uses: dorny/paths-filter@v2.11.1 - id: filter with: - list-files: shell - filters: | - changed: - - '**' - - name: Save changed files to env var - run: echo "CHANGED_FILES=${{ steps.filter.outputs.changed_files }}" >> $GITHUB_ENV - - - name: Parse changed files with python script - run: | - echo "PLUGIN_INFO=$(python .github/workflows/parse_changed_files.py '${{ env.CHANGED_FILES }}')" >> $GITHUB_ENV - - - name: Save plugin info to outputs - id: getpluginfo - run: | - echo "PLUGIN_INFO=$PLUGIN_INFO" >> $GITHUB_OUTPUT + fetch-depth: 0 - - name: check if scoring needed - id: runscore + - name: Save changed files to env var + run: echo "CHANGED_FILES=$(git diff --name-only origin/main origin/$GITHUB_HEAD_REF | tr '\n' ' ')" >> $GITHUB_ENV + + - name: Get scoring info run: | + python -m pip install . + echo "PLUGIN_INFO=$(python -c "from brainscore_core.plugin_management.parse_plugin_changes import get_scoring_info; get_scoring_info('${{ env.CHANGED_FILES }}', 'brainscore_language')" >> $GITHUB_OUTPUT echo "RUN_SCORE=$(jq -r '.run_score' <<< "$PLUGIN_INFO")" >> $GITHUB_OUTPUT + get_submitter_info: name: Get PR author email and (if web submission) Brain-Score user ID runs-on: ubuntu-latest @@ -92,6 +80,7 @@ jobs: run: | echo "The Brain-Score user ID is ${{ steps.getuid.outputs.BS_UID }}" echo "PLUGIN_INFO="$(<<<$PLUGIN_INFO jq '. + {user_id: ${{ steps.getuid.outputs.UID }} }')"" >> $GITHUB_ENV + - name: Get PR author email from GitHub username id: getemail uses: evvanErb/get-github-email-by-username-action@v1.25 @@ -104,6 +93,7 @@ jobs: echo "The PR author email is ${{ steps.getemail.outputs.email }}" echo "PLUGIN_INFO="$(<<<$PLUGIN_INFO jq '. + {author_email: "'${{ steps.getemail.outputs.email }}'"}')"" >> $GITHUB_OUTPUT + runscore: name: Score plugins runs-on: ubuntu-latest @@ -125,4 +115,4 @@ jobs: - name: Build project and run scoring run: | python -m pip install "." - python .github/workflows/call_endpoints.py call_jenkins '${{ env.PLUGIN_INFO }}' + python -c "from brainscore_language.submission.endpoints import call_jenkins; call_jenkins('$env.PLUGIN_INFO')" diff --git a/.github/workflows/travis_trigger.sh b/.github/workflows/travis_trigger.sh old mode 100644 new mode 100755 index 283aeb16..28258865 --- a/.github/workflows/travis_trigger.sh +++ b/.github/workflows/travis_trigger.sh @@ -2,9 +2,10 @@ GH_WORKFLOW_TRIGGER=$1 TRAVIS_PULL_REQUEST_SHA=$2 +PLUGIN_ONLY=$3 curl -L -X POST \ -H "Authorization: token $GH_WORKFLOW_TRIGGER" \ --d '{"state": "success", "description": "Run automerge workflow", +-d '{"state": "success", "description": "Run automerge workflow - '$PLUGIN_ONLY'", "context": "continuous-integration/travis"}' \ "https://api.github.com/repos/brain-score/language/statuses/$TRAVIS_PULL_REQUEST_SHA" \ No newline at end of file diff --git a/.readthedocs.yml b/.readthedocs.yml index d103d674..57a75f48 100644 --- a/.readthedocs.yml +++ b/.readthedocs.yml @@ -1,4 +1,10 @@ +version: 2 + +build: + os: "ubuntu-22.04" + tools: + python: "3.8" + python: - version: 3.8 install: - requirements: docs/requirements.txt diff --git a/.travis.yml b/.travis.yml index 883dd5c2..e89328e7 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,19 +1,10 @@ language: python -matrix: - include: - - name: 3.8 public - python: '3.8' - - name: 3.8 private - python: '3.8' - env: PRIVATE_ACCESS=1 - - name: 3.9 public - python: '3.9' - - name: 3.9 private - python: '3.9' - env: PRIVATE_ACCESS=1 env: global: - PYTEST_SETTINGS="not requires_gpu and not memory_intense and not slow and not travis_slow" + - MODIFIES_PLUGIN="False" + - PLUGIN_ONLY="False" + - WEB_SUBMISSION="False" install: - python -m pip install -e ".[test]" # install conda for plugin runner @@ -27,14 +18,43 @@ install: - pip list # install singularity for container models - conda install -yc conda-forge singularity +before_script: + - git fetch --depth=50 origin refs/heads/main:refs/heads/main script: - - if [ "$PRIVATE_ACCESS" = 1 ] && [ -n "${GITHUB_TOKEN}" ]; then pytest -m "private_access and $PYTEST_SETTINGS"; fi - - if [ "$PRIVATE_ACCESS" != 1 ]; then pytest -m "not private_access and $PYTEST_SETTINGS" --ignore "tests/test_submission"; fi - # if plugin files added or modified, run tests for those plugins - - if [ "$TRAVIS_PULL_REQUEST" = "true" ]; then python -c "from brainscore_core.plugin_management.parse_plugin_changes import run_changed_plugin_tests; run_changed_plugin_tests($TRAVIS_PULL_REQUEST_SHA, 'brainscore_language')"; fi + # if ONLY plugin files changed, ONLY run tests for those plugins; otherwise, run full test suite + - CHANGED_FILES=$( echo $(git diff --name-only main $TRAVIS_PULL_REQUEST_SHA -C $TRAVIS_BUILD_DIR) | tr '\n' ' ' ) + - | + if [ "$TRAVIS_PULL_REQUEST" ]; then + TESTING_NEEDED=$( python -c "from brainscore_core.plugin_management.parse_plugin_changes import get_testing_info; get_testing_info(\"${CHANGED_FILES}\", 'brainscore_language')" ) && + read MODIFIES_PLUGIN PLUGIN_ONLY <<< $TESTING_NEEDED && echo MODIFIES_PLUGIN: $MODIFIES_PLUGIN && echo PLUGIN_ONLY: $PLUGIN_ONLY; + fi + - if [ "$MODIFIES_PLUGIN" = "True" ]; then python -c "from brainscore_core.plugin_management.parse_plugin_changes import run_changed_plugin_tests; run_changed_plugin_tests(\"${CHANGED_FILES}\", 'brainscore_language')"; fi + - if [ "$PRIVATE_ACCESS" = 1 ] && [ -n "${GITHUB_TOKEN}" ] && [ "$PLUGIN_ONLY" = "False" ]; then pytest -m "private_access and $PYTEST_SETTINGS"; fi + - if [ "$PRIVATE_ACCESS" != 1 ] && [ "$PLUGIN_ONLY" = "False" ]; then pytest -m "not private_access and $PYTEST_SETTINGS" --ignore "tests/test_submission"; fi +after_failure: + # if web submission, notify submitter via email + - PR_TITLE=$(curl https://github.com/${TRAVIS_REPO_SLUG}/pull/${TRAVIS_PULL_REQUEST} 2> /dev/null | grep "title" | head -1) + - if [[ "$PR_TITLE" = "brain-score.org (user:"* ]]; then WEB_SUBMISSION="True"; fi + - | + if $WEB_SUBMISSION = "True"; then + BRAINSCORE_UID=$(<<<$PR_TITLE | sed -E 's/.*\(user:([^)]+)\).*/\1/') && + python -c "from brainscore_language.submission.endpoints import send_email_to_submitter; send_email_to_submitter(\"${BRAINSCORE_UID}\", 'language', \"${$TRAVIS_PULL_REQUEST}\", \"${$GMAIL_USERNAME}\", \"${GMAIL_PASSWORD}\")"; + fi + jobs: include: - # trigger workflow to check if plugin is being added - - stage: "Trigger automerge workflow" - script: if [ "$TRAVIS_PULL_REQUEST" = "true" ]; then bash .github/workflows/travis_trigger $GH_WORKFLOW_TRIGGER $TRAVIS_PULL_REQUEST_SHA; fi + - name: 3.8 public + python: '3.8' + - name: 3.8 private + python: '3.8' + env: PRIVATE_ACCESS=1 + - name: 3.9 public + python: '3.9' + - name: 3.9 private + python: '3.9' + env: PRIVATE_ACCESS=1 + - stage: "Automerge check" + install: skip + if: type = pull_request + script: if [ "$PLUGIN_ONLY" = "True" ]; then bash ${TRAVIS_BUILD_DIR}/.github/workflows/travis_trigger.sh $GH_WORKFLOW_TRIGGER $TRAVIS_PULL_REQUEST_SHA $PLUGIN_ONLY; fi diff --git a/brainscore_language/submission/endpoints.py b/brainscore_language/submission/endpoints.py index 73239f0c..8ebc82fd 100644 --- a/brainscore_language/submission/endpoints.py +++ b/brainscore_language/submission/endpoints.py @@ -2,6 +2,8 @@ import argparse from typing import List, Union, Dict, Tuple from distutils.util import strtobool +import requests +from requests.auth import HTTPBasicAuth from brainscore_core import Score, Benchmark from brainscore_core.submission import UserManager, RunScoringEndpoint, DomainPlugins @@ -22,9 +24,11 @@ def call_jenkins(plugin_info: Dict[str, Union[List[str], str]]): url = f'{jenkins_base}/job/{jenkins_job}/buildWithParameters?token={jenkins_trigger}' payload = {k: v for k, v in plugin_info.items() if plugin_info[k]} - auth_basic = HTTPBasicAuth(username=jenkins_user, password=jenkins_token) - r = requests.get(url, params=payload, auth=auth_basic) - logger.debug(r) + try: + auth_basic = HTTPBasicAuth(username=jenkins_user, password=jenkins_token) + r = requests.get(url, params=payload, auth=auth_basic) + except Exception as e: + print(f'Could not initiate Jenkins job because of {e}') class LanguagePlugins(DomainPlugins): @@ -42,15 +46,19 @@ def score(self, model_identifier: str, benchmark_identifier: str) -> Score: run_scoring_endpoint = RunScoringEndpoint(language_plugins, db_secret=config.get_database_secret()) -def get_user_email(uid: int) -> str: - """ Convenience method for GitHub Actions to get a user's email if their web-submitted PR fails. """ - return get_email_from_uid(uid) +def send_email_to_submitter(uid: int, domain: str, pr_number: str, + mail_username: str, mail_password:str ): + """ Send submitter an email if their web-submitted PR fails. """ + subject = "Brain-Score submission failed" + body = f"Your Brain-Score submission did not pass checks. Please review the test results and update the PR at https://github.com/brain-score/{domain}/pull/{pr_number} or send in an updated submission via the website." + user_manager = UserManager(db_secret=config.get_database_secret()) + return user_manager.send_user_email(uid, body, mail_username, mail_password) -def create_user(domain: str, email: str) -> int: - user_manager = UserManager(domain, email, db_secret=config.get_database_secret()) - new_user_id = user_manager() - return new_user_id +def get_user_id(email: str) -> int: + user_manager = UserManager(db_secret=config.get_database_secret()) + user_id = user_manager.get_uid(email) + return user_id def _get_ids(args_dict: Dict[str, Union[str, List]], key: str) -> Union[List, str, None]: @@ -141,4 +149,4 @@ def parse_args() -> argparse.Namespace: elif args.fn == 'get_models_and_benchmarks': get_models_and_benchmarks(args_dict) else: - raise ValueError(f'Invalid method: {args.fn}') + raise ValueError(f'Invalid method: {args.fn}') \ No newline at end of file diff --git a/docs/requirements.txt b/docs/requirements.txt index 5fc89540..23532fd2 100644 --- a/docs/requirements.txt +++ b/docs/requirements.txt @@ -2,3 +2,4 @@ Sphinx>=4 sphinx_rtd_theme +recommonmark \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml index 3d3b5ba3..a91228a7 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -22,6 +22,8 @@ dependencies = [ "transformers>=4.11.3", "gensim", "joblib", + # submission dependencies + "requests" ] [project.optional-dependencies]