diff --git a/.github/workflows/python-tests.yml b/.github/workflows/python-tests.yml index a50ccc382..d88f68262 100644 --- a/.github/workflows/python-tests.yml +++ b/.github/workflows/python-tests.yml @@ -1,14 +1,14 @@ name: Python tests -on: [push, workflow_dispatch] +on: [ push, workflow_dispatch ] jobs: test: runs-on: ubuntu-22.04 strategy: - fail-fast: false matrix: - python-version: ["3.10.12"] + python-version: [ "3.10.12" ] + poetry-version: [ "1.8.3" ] # Service containers to run with `test` services: @@ -23,10 +23,6 @@ jobs: POSTGRES_PASSWORD: password # Set health checks to wait until postgres has started options: >- - --health-cmd pg_isready - --health-interval 10s - --health-timeout 5s - --health-retries 5 --name postgres ports: # Maps tcp port 5432 on service container to the host @@ -38,68 +34,63 @@ jobs: # Set health checks to wait until redis has started options: >- --health-cmd "redis-cli ping" - --health-interval 10s + --health-interval 5s --health-timeout 5s --health-retries 5 ports: # Maps tcp port 6379 on service container to the host - 6379:6379 steps: - - name: Increase max_connections - run: | - docker exec -i postgres bash << EOF - sed -i -e 's/max_connections = 100/max_connections = 10000/' /var/lib/postgresql/data/postgresql.conf - EOF - - name: Restart postgres - run: | - docker restart --time 0 postgres && sleep 5 - - - uses: actions/checkout@v4 -# with: -# submodules: recursive + # https://remarkablemark.org/blog/2022/05/12/github-actions-postgresql-increase-max-connections-and-shared-buffers/ + - name: Increase max_connections + run: >- + docker exec -i postgres bash << EOF + sed -i -e 's/max_connections = 100/max_connections = 10000/' /var/lib/postgresql/data/postgresql.conf + EOF + - name: Restart postgres + run: >- + docker restart postgres + && while ! docker exec postgres pg_isready; do sleep 5; done - - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v5 - with: - python-version: ${{ matrix.python-version }} - cache: 'pip' - - - name: Install system dependencies - run: | - sudo apt-get update && sudo apt-get install -y --no-install-recommends \ - libpoppler-cpp-dev \ - python3-opencv \ - postgresql-client \ + - name: Install system dependencies + run: >- + sudo apt-get update && sudo apt-get install -y --no-install-recommends + libpoppler-cpp-dev + python3-opencv + postgresql-client libzbar0 - - name: Install python dependencies - run: | - pip install -U poetry pip && poetry install --only main --no-interaction + - uses: actions/checkout@v4 -# - name: Load secrets into env -# uses: oNaiPs/secrets-to-env-action@v1 -# with: -# secrets: ${{ toJSON(secrets) }} + - name: Setup Python, Poetry and Dependencies + uses: packetcoders/action-setup-cache-python-poetry@main + with: + python-version: ${{matrix.python-version}} + poetry-version: ${{matrix.poetry-version}} + install-args: --only main - - name: Test with pytest - env: - PGHOST: localhost - PGPORT: 5432 - PGDATABASE: gooey - PGUSER: postgres - PGPASSWORD: password - REDIS_URL: redis://localhost:6379/0 - REDIS_CACHE_URL: redis://localhost:6379/1 - APP_BASE_URL: http://localhost:3000 - API_BASE_URL: http://localhost:8080 - ADMIN_BASE_URL: http://localhost:8000 - GOOGLE_APPLICATION_CREDENTIALS_JSON: ${{ secrets.GOOGLE_APPLICATION_CREDENTIALS_JSON }} - GS_BUCKET_NAME: ${{ secrets.GS_BUCKET_NAME }} - STRIPE_SECRET_KEY: ${{ secrets.STRIPE_SECRET_KEY }} - ELEVEN_LABS_API_KEY: ${{ secrets.ELEVEN_LABS_API_KEY }} - AZURE_SPEECH_REGION: ${{ secrets.AZURE_SPEECH_REGION }} - AZURE_SPEECH_KEY: ${{ secrets.AZURE_SPEECH_KEY }} - AZURE_FORM_RECOGNIZER_ENDPOINT: ${{ secrets.AZURE_FORM_RECOGNIZER_ENDPOINT }} - AZURE_FORM_RECOGNIZER_KEY: ${{ secrets.AZURE_FORM_RECOGNIZER_KEY }} - run: | - poetry run ./scripts/run-tests.sh + - name: Run tests + env: + PGHOST: localhost + PGPORT: 5432 + PGDATABASE: gooey + PGUSER: postgres + PGPASSWORD: password + REDIS_URL: redis://localhost:6379/0 + REDIS_CACHE_URL: redis://localhost:6379/1 + APP_BASE_URL: http://localhost:3000 + API_BASE_URL: http://localhost:8080 + ADMIN_BASE_URL: http://localhost:8000 + GOOGLE_APPLICATION_CREDENTIALS_JSON: ${{ secrets.GOOGLE_APPLICATION_CREDENTIALS_JSON }} + GS_BUCKET_NAME: ${{ secrets.GS_BUCKET_NAME }} + STRIPE_SECRET_KEY: ${{ secrets.STRIPE_SECRET_KEY }} + ELEVEN_LABS_API_KEY: ${{ secrets.ELEVEN_LABS_API_KEY }} + AZURE_SPEECH_REGION: ${{ secrets.AZURE_SPEECH_REGION }} + AZURE_SPEECH_KEY: ${{ secrets.AZURE_SPEECH_KEY }} + AZURE_FORM_RECOGNIZER_ENDPOINT: ${{ secrets.AZURE_FORM_RECOGNIZER_ENDPOINT }} + AZURE_FORM_RECOGNIZER_KEY: ${{ secrets.AZURE_FORM_RECOGNIZER_KEY }} + TEST_SLACK_TEAM_ID: ${{ secrets.TEST_SLACK_TEAM_ID }} + TEST_SLACK_USER_ID: ${{ secrets.TEST_SLACK_USER_ID }} + TEST_SLACK_AUTH_TOKEN: ${{ secrets.TEST_SLACK_AUTH_TOKEN }} + run: | + poetry run ./scripts/run-tests.sh diff --git a/README.md b/README.md index b2b487b11..ca0f30cb8 100644 --- a/README.md +++ b/README.md @@ -31,7 +31,8 @@ - Cloud Speech Administrator - Cloud Translation API Admin - Firebase Authentication Admin -5. Download the `serviceAccountKey.json` and save it to the project root. + - Storage Admin +5. Create and Download a JSON Key for this service account and save it to the project root as `serviceAccountKey.json`. * Run tests to see if everything is working fine: ``` diff --git a/daras_ai_v2/asr.py b/daras_ai_v2/asr.py index f46de107c..057092d75 100644 --- a/daras_ai_v2/asr.py +++ b/daras_ai_v2/asr.py @@ -1,6 +1,7 @@ import os.path import os.path import tempfile +import typing from enum import Enum import requests @@ -36,6 +37,7 @@ SHORT_FILE_CUTOFF = 5 * 1024 * 1024 # 1 MB +# https://cloud.google.com/translate/docs/languages#roman TRANSLITERATION_SUPPORTED = {"ar", "bn", " gu", "hi", "ja", "kn", "ru", "ta", "te"} # https://cloud.google.com/speech-to-text/docs/speech-to-text-supported-languages @@ -395,15 +397,6 @@ def google_translate_source_languages() -> dict[str, str]: } -def get_language_in_collection(langcode: str, languages): - import langcodes - - for lang in languages: - if langcodes.get(lang).language == langcodes.get(langcode).language: - return langcode - return None - - def asr_language_selector( selected_model: AsrModels, label="##### Spoken Language", @@ -484,26 +477,17 @@ def run_ghana_nlp_translate( target_language: str, source_language: str, ) -> list[str]: - import langcodes - - assert ( - target_language in GHANA_NLP_SUPPORTED - ), "Ghana NLP does not support this target language" - assert source_language, "Source language is required for Ghana NLP" - - if source_language not in GHANA_NLP_SUPPORTED: - src = langcodes.Language.get(source_language).language - for lang in GHANA_NLP_SUPPORTED: - if src == langcodes.Language.get(lang).language: - source_language = lang - break assert ( - source_language in GHANA_NLP_SUPPORTED - ), "Ghana NLP does not support this source language" - + source_language and target_language + ), "Both Source & Target language is required for Ghana NLP" + source_language = normalised_lang_in_collection( + source_language, GHANA_NLP_SUPPORTED + ) + target_language = normalised_lang_in_collection( + target_language, GHANA_NLP_SUPPORTED + ) if source_language == target_language: return texts - return map_parallel( lambda doc: _call_ghana_nlp_chunked(doc, source_language, target_language), texts, @@ -550,50 +534,67 @@ def run_google_translate( list[str]: Translated text. """ from google.cloud import translate_v2 as translate - import langcodes - # convert to BCP-47 format (google handles consistent language codes but sometimes gets confused by a mix of iso2 and iso3 which we have) + supported_languages = google_translate_target_languages() if source_language: - source_language = langcodes.Language.get(source_language).to_tag() - source_language = get_language_in_collection( - source_language, google_translate_source_languages().keys() - ) # this will default to autodetect if language is not found as supported - target_language = langcodes.Language.get(target_language).to_tag() - target_language: str | None = get_language_in_collection( - target_language, google_translate_target_languages().keys() + try: + source_language = normalised_lang_in_collection( + source_language, supported_languages + ) + except UserError: + source_language = None # autodetect + target_language = normalised_lang_in_collection( + target_language, supported_languages ) - if not target_language: - raise UserError(f"Unsupported target language: {target_language!r}") # if the language supports transliteration, we should check if the script is Latin if source_language and source_language not in TRANSLITERATION_SUPPORTED: - language_codes = [source_language] * len(texts) + detected_source_languges = [source_language] * len(texts) else: translate_client = translate.Client() detections = flatten( translate_client.detect_language(texts[i : i + TRANSLATE_BATCH_SIZE]) for i in range(0, len(texts), TRANSLATE_BATCH_SIZE) ) - language_codes = [detection["language"] for detection in detections] + detected_source_languges = [detection["language"] for detection in detections] + + # fix for when sometimes google might detect a different language than the user provided one + if source_language: + detected_source_languges = [ + code if source_language in code.split("-")[0] else source_language + for code in detected_source_languges + ] return map_parallel( - lambda text, source: _translate_text( - text, source, target_language, glossary_url + lambda text, src_lang: _translate_text( + text, target_language, src_lang, glossary_url ), texts, - language_codes, + detected_source_languges, max_workers=TRANSLATE_BATCH_SIZE, ) +def normalised_lang_in_collection(target: str, collection: typing.Iterable[str]) -> str: + import langcodes + + for candidate in collection: + if langcodes.get(candidate).language == langcodes.get(target).language: + return candidate + + raise UserError( + f"Unsupported language: {target!r} | must be one of {set(collection)}" + ) + + def _translate_text( text: str, - source_language: str, target_language: str, + source_language: str, glossary_url: str | None, ) -> str: is_romanized = source_language.endswith("-Latn") - source_language = source_language.replace("-Latn", "") + source_language = source_language.split("-")[0] enable_transliteration = ( is_romanized and source_language in TRANSLITERATION_SUPPORTED ) @@ -602,9 +603,6 @@ def _translate_text( if not text or source_language == target_language or source_language == "und": return text - if source_language == "wo-SN" or target_language == "wo-SN": - return _MinT_translate_one_text(text, source_language, target_language) - config = { "target_language_code": target_language, "contents": text, @@ -614,7 +612,6 @@ def _translate_text( if source_language != "auto": config["source_language_code"] = source_language - # glossary does not work with transliteration if glossary_url and not enable_transliteration: from glossary_resources.models import GlossaryResource diff --git a/glossary_resources/tests.py b/glossary_resources/tests.py index 362e3c759..4a410b866 100644 --- a/glossary_resources/tests.py +++ b/glossary_resources/tests.py @@ -4,7 +4,7 @@ from daras_ai_v2 import settings from daras_ai_v2.crypto import get_random_doc_id from glossary_resources.models import GlossaryResource -from tests.test_translation import _test_run_google_translate_one +from tests.test_translation import google_translate_check GLOSSARY = [ { @@ -27,12 +27,18 @@ "pos": "noun", "description": "well labs agniastra", }, + { + "en-US": "Jalapeño", + "hi-IN": "मिर्ची", + "pos": "noun", + "description": "Jalapeño", + }, ] TRANSLATION_TESTS_GLOSSARY = [ ( - "एक एकड़ भूमि के लिए कितनी अग्निअस्त्र की आवश्यकता होती है", - "how many fire extinguishers are required for one acre of land", # default + "एक एकड़ भूमि के लिए कितनी अग्निअस्त्र की आवश्यकता होती है", # source + "how many fire extinguishers are required for one acre of land", # default translation "how many agniastra are required for one acre of land", # using glossary ), ( @@ -45,6 +51,11 @@ "What can we do with AI", "What can we do with Gooey.AI", ), + ( + "मेरे मिर्ची पर लाल धब्बे आ गये हैं", + "My chillies have got red spots", + "My Jalapeño have got red spots", + ), ] @@ -65,15 +76,15 @@ def glossary_url(): @pytest.mark.skipif(not settings.GS_BUCKET_NAME, reason="No GCS bucket") @pytest.mark.django_db -def test_run_google_translate_glossary(glossary_url, threadpool_subtest): +def test_google_translate_glossary(glossary_url, threadpool_subtest): for text, expected, expected_with_glossary in TRANSLATION_TESTS_GLOSSARY: threadpool_subtest( - _test_run_google_translate_one, + google_translate_check, text, expected, ) threadpool_subtest( - _test_run_google_translate_one, + google_translate_check, text, expected_with_glossary, glossary_url=glossary_url, diff --git a/tests/test_translation.py b/tests/test_translation.py index d3bf7318a..dcf7a2568 100644 --- a/tests/test_translation.py +++ b/tests/test_translation.py @@ -5,51 +5,68 @@ TRANSLATION_TESTS = [ # hindi romanized ( + "hi", "Hi Sir Mera khet me mircha ke ped me fal gal Kar gir hai to iske liye ham kon sa dawa de please help me", "hi sir in my field the fruits of chilli tree are rotting and falling so which medicine should i give for this please help", ), ( + "hi", "Mirchi ka ped", - "pepper tree", + "chilli tree", ), # telugu ( + "te", "90 రోజుల తర్వాత మిర్చి తోటలో ఏమేమి పోషకాలు వేసి వేయాలి", "after 90 days what nutrients should be added to the pepper garden?", ), # swahili ( + "sw", "Unastahili kuchanganya mchanga na nini unapopanda kahawa?", "What should you mix sand with when planting coffee?", ), # amharic ( + "am", "ለዘር የሚሆን የስንዴ ምርጥ ዘር ዓይነት ስንት ናቸው?እንደ ሀገረችን እትዮጵያ ደረጃ?", "What are the best types of wheat for seed? According to our country, Ethiopia?", ), + # spanish + ( + "es", + "hola senor me gusta el chile", + "hello sir, i like chili", + ), # english ( + "en", "what is the best type of wheat for seed?", "what is the best type of wheat for seed?", ), - ( - "hola senor me gusta el chile", - "hello sir, i like chili", - ), ] -def test_run_google_translate(threadpool_subtest): - for text, expected in TRANSLATION_TESTS: - threadpool_subtest(_test_run_google_translate_one, text, expected) +def test_google_translate(threadpool_subtest): + for lang, text, expected in TRANSLATION_TESTS: + threadpool_subtest(google_translate_check, text, expected, source_language=lang) @flaky -def _test_run_google_translate_one( - text: str, expected: str, glossary_url=None, target_lang="en" +def google_translate_check( + text: str, + expected: str, + *, + glossary_url: str = None, + target_language: str = "en", + source_language: str = None ): - actual = run_google_translate([text], target_lang, glossary_url=glossary_url)[0] - assert ( - actual.replace(".", "").replace(",", "").strip().lower() - == expected.replace(".", "").replace(",", "").strip().lower() - ) + actual = run_google_translate( + texts=[text], + target_language=target_language, + source_language=source_language, + glossary_url=glossary_url, + )[0] + actual_norm = actual.replace(".", "").replace(",", "").strip().lower() + expected_norm = expected.replace(".", "").replace(",", "").strip().lower() + assert actual_norm == expected_norm