From ae05c9fcc737e7d44024640ce8ad8b109e326e38 Mon Sep 17 00:00:00 2001 From: Andrew Date: Thu, 17 Oct 2024 11:19:55 -0300 Subject: [PATCH] Add support for RTL text direction for template content (#2321) * feat(rtl): add new text direction column to db, model * feat(rtl): add `text_direction_rtl` property to `TemplateSchema` * feat(rtl): add `text_direction_rtl` property to the list of properties checked to see if a template changed * chore: formatting * tests(rtl): add tests to ensure template updates when text direction changes * chore: update utils dep * chore: update poetry lock file * test(rtl): add tests for creating templates * chore: bump utils * chore: update poetry.lock --- app/models.py | 1 + app/schemas.py | 2 + app/template/rest.py | 11 +- .../versions/0461_add_rtl_column_templates.py | 26 +++ poetry.lock | 201 +----------------- pyproject.toml | 2 +- tests/app/db.py | 2 + tests/app/template/test_rest.py | 82 +++++++ 8 files changed, 132 insertions(+), 195 deletions(-) create mode 100644 migrations/versions/0461_add_rtl_column_templates.py diff --git a/app/models.py b/app/models.py index 8f06e860b8..6cbd0723fb 100644 --- a/app/models.py +++ b/app/models.py @@ -1096,6 +1096,7 @@ def __init__(self, **kwargs): hidden = db.Column(db.Boolean, nullable=False, default=False) subject = db.Column(db.Text) postage = db.Column(db.String, nullable=True) + text_direction_rtl = db.Column(db.Boolean, nullable=False, default=False) CheckConstraint( """ CASE WHEN template_type = 'letter' THEN diff --git a/app/schemas.py b/app/schemas.py index f588dd549e..b39f471776 100644 --- a/app/schemas.py +++ b/app/schemas.py @@ -406,6 +406,7 @@ class TemplateSchema(BaseTemplateSchema): redact_personalisation = fields.Method("redact") created_at = FlexibleDateTime() updated_at = FlexibleDateTime() + text_direction_rtl = field_for(models.Template, "text_direction_rtl") def get_is_precompiled_letter(self, template): return template.is_precompiled_letter @@ -578,6 +579,7 @@ class Meta(BaseSchema.Meta): "subject", "redact_personalisation", "is_precompiled_letter", + "text_direction_rtl", ], dump_only=True, ) diff --git a/app/template/rest.py b/app/template/rest.py index f70dfc5165..257a9ca218 100644 --- a/app/template/rest.py +++ b/app/template/rest.py @@ -278,7 +278,16 @@ def _template_has_not_changed(current_data, updated_template): return False return all( current_data[key] == updated_template[key] - for key in ("name", "content", "subject", "archived", "process_type", "postage", "template_category_id") + for key in ( + "name", + "content", + "subject", + "archived", + "process_type", + "postage", + "template_category_id", + "text_direction_rtl", + ) ) diff --git a/migrations/versions/0461_add_rtl_column_templates.py b/migrations/versions/0461_add_rtl_column_templates.py new file mode 100644 index 0000000000..e89d86f503 --- /dev/null +++ b/migrations/versions/0461_add_rtl_column_templates.py @@ -0,0 +1,26 @@ +""" +Revision ID: 0461_add_rtl_column_templates +Revises: 0460_new_service_columns +Create Date: 2024-06-27 13:32:00 +""" +import sqlalchemy as sa +from alembic import op + +revision = "0461_add_rtl_column_templates" +down_revision = "0460_new_service_columns" + + +def upgrade(): + # Add the new column to the templates table + op.add_column("templates", sa.Column("text_direction_rtl", sa.Boolean(), nullable=False, server_default=sa.false())) + + # Add the new column to the templates_history table + op.add_column("templates_history", sa.Column("text_direction_rtl", sa.Boolean(), nullable=False, server_default=sa.false())) + + +def downgrade(): + # Remove the column from the templates table + op.drop_column("templates", "text_direction_rtl") + + # Remove the column from the templates_history table + op.drop_column("templates_history", "text_direction_rtl") diff --git a/poetry.lock b/poetry.lock index 470dd236e1..036a3d0e95 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,10 +1,9 @@ -# This file is automatically @generated by Poetry and should not be changed by hand. +# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. [[package]] name = "aiohappyeyeballs" version = "2.4.0" description = "Happy Eyeballs for asyncio" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -16,7 +15,6 @@ files = [ name = "aiohttp" version = "3.10.5" description = "Async http client/server framework (asyncio)" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -129,7 +127,6 @@ speedups = ["Brotli", "aiodns (>=3.2.0)", "brotlicffi"] name = "aiosignal" version = "1.3.1" description = "aiosignal: a list of registered asynchronous callbacks" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -144,7 +141,6 @@ frozenlist = ">=1.1.0" name = "alembic" version = "1.13.2" description = "A database migration tool for SQLAlchemy." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -164,7 +160,6 @@ tz = ["backports.zoneinfo"] name = "amqp" version = "5.2.0" description = "Low-level AMQP client for Python (fork of amqplib)." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -179,7 +174,6 @@ vine = ">=5.0.0,<6.0.0" name = "annotated-types" version = "0.7.0" description = "Reusable constraint types to use with typing.Annotated" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -191,7 +185,6 @@ files = [ name = "apig-wsgi" version = "2.18.0" description = "Wrap a WSGI application in an AWS Lambda handler function for running on API Gateway or an ALB." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -203,7 +196,6 @@ files = [ name = "async-timeout" version = "4.0.3" description = "Timeout context manager for asyncio programs" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -215,7 +207,6 @@ files = [ name = "attrs" version = "24.2.0" description = "Classes Without Boilerplate" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -235,7 +226,6 @@ tests-mypy = ["mypy (>=1.11.1)", "pytest-mypy-plugins"] name = "aws-embedded-metrics" version = "1.0.8" description = "AWS Embedded Metrics Package" -category = "main" optional = false python-versions = "*" files = [ @@ -250,7 +240,6 @@ aiohttp = "*" name = "aws-xray-sdk" version = "2.14.0" description = "The AWS X-Ray SDK for Python (the SDK) enables Python developers to record and emit information from within their applications to the AWS X-Ray service." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -266,7 +255,6 @@ wrapt = "*" name = "awscli" version = "1.33.35" description = "Universal Command Line Environment for AWS." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -286,7 +274,6 @@ s3transfer = ">=0.10.0,<0.11.0" name = "awscli-cwlogs" version = "1.4.6" description = "AWSCLI CloudWatch Logs plugin" -category = "main" optional = false python-versions = "*" files = [ @@ -303,7 +290,6 @@ six = ">=1.1.0" name = "bcrypt" version = "4.2.0" description = "Modern password hashing for your software and your servers" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -344,7 +330,6 @@ typecheck = ["mypy"] name = "billiard" version = "4.2.0" description = "Python multiprocessing fork with improvements and bugfixes" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -356,7 +341,6 @@ files = [ name = "black" version = "23.12.1" description = "The uncompromising code formatter." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -403,7 +387,6 @@ uvloop = ["uvloop (>=0.15.2)"] name = "bleach" version = "6.1.0" description = "An easy safelist-based HTML-sanitizing tool." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -422,7 +405,6 @@ css = ["tinycss2 (>=1.1.0,<1.3)"] name = "blinker" version = "1.8.2" description = "Fast, simple object-to-object and broadcast signaling" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -434,7 +416,6 @@ files = [ name = "boto" version = "2.49.0" description = "Amazon Web Services Library" -category = "main" optional = false python-versions = "*" files = [ @@ -446,7 +427,6 @@ files = [ name = "boto3" version = "1.34.153" description = "The AWS SDK for Python" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -466,7 +446,6 @@ crt = ["botocore[crt] (>=1.21.0,<2.0a0)"] name = "botocore" version = "1.34.153" description = "Low-level, data-driven core of boto 3." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -486,7 +465,6 @@ crt = ["awscrt (==0.20.11)"] name = "brotli" version = "1.1.0" description = "Python bindings for the Brotli compression library" -category = "dev" optional = false python-versions = "*" files = [ @@ -579,7 +557,6 @@ files = [ name = "cachelib" version = "0.12.0" description = "A collection of cache libraries in the same API interface." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -591,7 +568,6 @@ files = [ name = "cachetools" version = "4.2.4" description = "Extensible memoizing collections and decorators" -category = "main" optional = false python-versions = "~=3.5" files = [ @@ -603,7 +579,6 @@ files = [ name = "celery" version = "5.4.0" description = "Distributed Task Queue." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -666,7 +641,6 @@ zstd = ["zstandard (==0.22.0)"] name = "certifi" version = "2024.8.30" description = "Python package for providing Mozilla's CA Bundle." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -678,7 +652,6 @@ files = [ name = "cffi" version = "1.17.0" description = "Foreign Function Interface for Python calling C code." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -758,7 +731,6 @@ pycparser = "*" name = "cfgv" version = "3.4.0" description = "Validate configuration and produce human readable error messages." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -770,7 +742,6 @@ files = [ name = "charset-normalizer" version = "3.3.2" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." -category = "main" optional = false python-versions = ">=3.7.0" files = [ @@ -870,7 +841,6 @@ files = [ name = "click" version = "8.1.7" description = "Composable command line interface toolkit" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -885,7 +855,6 @@ colorama = {version = "*", markers = "platform_system == \"Windows\""} name = "click-datetime" version = "0.2" description = "Datetime type support for click." -category = "main" optional = false python-versions = "*" files = [ @@ -903,7 +872,6 @@ dev = ["wheel"] name = "click-didyoumean" version = "0.3.1" description = "Enables git-like *did-you-mean* feature in click" -category = "main" optional = false python-versions = ">=3.6.2" files = [ @@ -918,7 +886,6 @@ click = ">=7" name = "click-plugins" version = "1.1.1" description = "An extension module for click to enable registering CLI commands via setuptools entry-points." -category = "main" optional = false python-versions = "*" files = [ @@ -936,7 +903,6 @@ dev = ["coveralls", "pytest (>=3.6)", "pytest-cov", "wheel"] name = "click-repl" version = "0.3.0" description = "REPL plugin for Click" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -955,7 +921,6 @@ testing = ["pytest (>=7.2.1)", "pytest-cov (>=4.0.0)", "tox (>=4.4.3)"] name = "colorama" version = "0.4.6" description = "Cross-platform colored terminal text." -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" files = [ @@ -967,7 +932,6 @@ files = [ name = "configargparse" version = "1.7" description = "A drop-in replacement for argparse that allows options to also be set via config files and/or environment variables." -category = "dev" optional = false python-versions = ">=3.5" files = [ @@ -983,7 +947,6 @@ yaml = ["PyYAML"] name = "coverage" version = "6.5.0" description = "Code coverage measurement for Python" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1049,7 +1012,6 @@ toml = ["tomli"] name = "coveralls" version = "3.3.1" description = "Show coverage stats online via coveralls.io" -category = "dev" optional = false python-versions = ">= 3.5" files = [ @@ -1058,7 +1020,7 @@ files = [ ] [package.dependencies] -coverage = ">=4.1,<6.0.0 || >6.1,<6.1.1 || >6.1.1,<7.0" +coverage = ">=4.1,<6.0.dev0 || >6.1,<6.1.1 || >6.1.1,<7.0" docopt = ">=0.6.1" requests = ">=1.0.0" @@ -1069,7 +1031,6 @@ yaml = ["PyYAML (>=3.10)"] name = "cryptography" version = "43.0.1" description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -1119,7 +1080,6 @@ test-randomorder = ["pytest-randomly"] name = "distlib" version = "0.3.8" description = "Distribution utilities" -category = "main" optional = false python-versions = "*" files = [ @@ -1131,7 +1091,6 @@ files = [ name = "docopt" version = "0.6.2" description = "Pythonic argument parser, that will make you smile" -category = "main" optional = false python-versions = "*" files = [ @@ -1142,7 +1101,6 @@ files = [ name = "docutils" version = "0.16" description = "Docutils -- Python Documentation Utilities" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -1154,7 +1112,6 @@ files = [ name = "environs" version = "9.5.0" description = "simplified environment variable parsing" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -1176,7 +1133,6 @@ tests = ["dj-database-url", "dj-email-url", "django-cache-url", "pytest"] name = "exceptiongroup" version = "1.2.2" description = "Backport of PEP 654 (exception groups)" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1191,7 +1147,6 @@ test = ["pytest (>=6)"] name = "execnet" version = "2.1.1" description = "execnet: rapid multi-Python deployment" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -1206,7 +1161,6 @@ testing = ["hatch", "pre-commit", "pytest", "tox"] name = "fido2" version = "0.9.3" description = "Python based FIDO 2.0 library" -category = "main" optional = false python-versions = ">=2.7.6,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*" files = [ @@ -1224,7 +1178,6 @@ pcsc = ["pyscard"] name = "filelock" version = "3.15.4" description = "A platform independent file lock." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1241,7 +1194,6 @@ typing = ["typing-extensions (>=4.8)"] name = "flake8" version = "6.1.0" description = "the modular source code checker: pep8 pyflakes and co" -category = "dev" optional = false python-versions = ">=3.8.1" files = [ @@ -1258,7 +1210,6 @@ pyflakes = ">=3.1.0,<3.2.0" name = "flask" version = "2.3.3" description = "A simple framework for building complex web applications." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1281,7 +1232,6 @@ dotenv = ["python-dotenv"] name = "flask-bcrypt" version = "1.0.1" description = "Brcrypt hashing for Flask." -category = "main" optional = false python-versions = "*" files = [ @@ -1297,7 +1247,6 @@ Flask = "*" name = "flask-cors" version = "4.0.1" description = "A Flask extension adding a decorator for CORS support" -category = "dev" optional = false python-versions = "*" files = [ @@ -1312,7 +1261,6 @@ Flask = ">=0.9" name = "flask-login" version = "0.6.3" description = "User authentication and session management for Flask." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1328,7 +1276,6 @@ Werkzeug = ">=1.0.1" name = "flask-marshmallow" version = "0.14.0" description = "Flask + marshmallow for beautiful APIs" -category = "main" optional = false python-versions = "*" files = [ @@ -1352,7 +1299,6 @@ tests = ["flask-sqlalchemy", "marshmallow-sqlalchemy (>=0.13.0)", "marshmallow-s name = "flask-migrate" version = "2.7.0" description = "SQLAlchemy database migrations for Flask applications using Alembic" -category = "main" optional = false python-versions = "*" files = [ @@ -1369,7 +1315,6 @@ Flask-SQLAlchemy = ">=1.0" name = "flask-redis" version = "0.4.0" description = "A nice way to use Redis in your Flask app" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -1389,7 +1334,6 @@ tests = ["coverage", "pytest", "pytest-mock"] name = "Flask-SQLAlchemy" version = "2.3.2" description = "Adds SQLAlchemy support to your Flask application" -category = "main" optional = false python-versions = "*" files = [] @@ -1409,7 +1353,6 @@ resolved_reference = "500e732dd1b975a56ab06a46bd1a20a21e682262" name = "freezegun" version = "1.5.1" description = "Let your Python tests travel through time" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1424,7 +1367,6 @@ python-dateutil = ">=2.7" name = "frozenlist" version = "1.4.1" description = "A list-like structure which implements collections.abc.MutableSequence" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1511,7 +1453,6 @@ files = [ name = "future" version = "1.0.0" description = "Clean single-source support for Python 3 and 2" -category = "main" optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -1523,7 +1464,6 @@ files = [ name = "gevent" version = "23.9.1" description = "Coroutine-based network library" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1586,7 +1526,6 @@ test = ["cffi (>=1.12.2)", "coverage (>=5.0)", "dnspython (>=1.16.0,<2.0)", "idn name = "geventhttpclient" version = "2.3.1" description = "HTTP client library for gevent" -category = "dev" optional = false python-versions = ">=3.9" files = [ @@ -1670,7 +1609,6 @@ examples = ["oauth2"] name = "greenlet" version = "2.0.2" description = "Lightweight in-process concurrent programming" -category = "main" optional = false python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" files = [ @@ -1748,7 +1686,6 @@ test = ["objgraph", "psutil"] name = "gunicorn" version = "20.1.0" description = "WSGI HTTP Server for UNIX" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -1769,7 +1706,6 @@ tornado = ["tornado (>=0.2)"] name = "identify" version = "2.6.0" description = "File identification library for Python" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1784,7 +1720,6 @@ license = ["ukkonen"] name = "idna" version = "2.10" description = "Internationalized Domain Names in Applications (IDNA)" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -1796,7 +1731,6 @@ files = [ name = "iniconfig" version = "2.0.0" description = "brain-dead simple config-ini parsing" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -1808,7 +1742,6 @@ files = [ name = "iso8601" version = "2.1.0" description = "Simple module to parse ISO 8601 dates" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -1820,7 +1753,6 @@ files = [ name = "isodate" version = "0.6.1" description = "An ISO 8601 date/time/duration parser and formatter" -category = "main" optional = false python-versions = "*" files = [ @@ -1835,7 +1767,6 @@ six = "*" name = "isort" version = "5.13.2" description = "A Python utility / library to sort Python imports." -category = "dev" optional = false python-versions = ">=3.8.0" files = [ @@ -1850,7 +1781,6 @@ colors = ["colorama (>=0.4.6)"] name = "itsdangerous" version = "2.2.0" description = "Safely pass data to untrusted environments and back." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1862,7 +1792,6 @@ files = [ name = "jinja2" version = "3.1.4" description = "A very fast and expressive template engine." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -1880,7 +1809,6 @@ i18n = ["Babel (>=2.7)"] name = "jinja2-cli" version = "0.8.2" description = "A CLI interface to Jinja2" -category = "dev" optional = false python-versions = "*" files = [ @@ -1902,7 +1830,6 @@ yaml = ["jinja2", "pyyaml"] name = "jmespath" version = "1.0.1" description = "JSON Matching Expressions" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -1914,7 +1841,6 @@ files = [ name = "jsonschema" version = "3.2.0" description = "An implementation of JSON Schema validation for Python" -category = "main" optional = false python-versions = "*" files = [ @@ -1936,7 +1862,6 @@ format-nongpl = ["idna", "jsonpointer (>1.13)", "rfc3339-validator", "rfc3986-va name = "kombu" version = "5.4.0" description = "Messaging library for Python." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -1972,7 +1897,6 @@ zookeeper = ["kazoo (>=2.8.0)"] name = "locust" version = "2.31.3" description = "Developer-friendly load testing framework" -category = "dev" optional = false python-versions = ">=3.9" files = [ @@ -2000,7 +1924,6 @@ Werkzeug = ">=2.0.0" name = "lxml" version = "5.3.0" description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -2155,7 +2078,6 @@ source = ["Cython (>=3.0.11)"] name = "mako" version = "1.3.5" description = "A super-fast templating language that borrows the best ideas from the existing templating languages." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2175,7 +2097,6 @@ testing = ["pytest"] name = "markdown-it-py" version = "3.0.0" description = "Python port of markdown-it. Markdown parsing, done right!" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2200,7 +2121,6 @@ testing = ["coverage", "pytest", "pytest-cov", "pytest-regressions"] name = "markupsafe" version = "2.1.5" description = "Safely add untrusted strings to HTML/XML markup." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2270,7 +2190,6 @@ files = [ name = "marshmallow" version = "3.22.0" description = "A lightweight library for converting complex datatypes to and from native Python datatypes." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2290,7 +2209,6 @@ tests = ["pytest", "pytz", "simplejson"] name = "marshmallow-sqlalchemy" version = "0.30.0" description = "SQLAlchemy integration with the marshmallow (de)serialization library" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2313,7 +2231,6 @@ tests = ["pytest", "pytest-lazy-fixture (>=0.6.2)"] name = "mccabe" version = "0.7.0" description = "McCabe checker, plugin for flake8" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -2325,7 +2242,6 @@ files = [ name = "mdurl" version = "0.1.2" description = "Markdown URL utilities" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2337,7 +2253,6 @@ files = [ name = "mistune" version = "0.8.4" description = "The fastest markdown parser in pure Python" -category = "main" optional = false python-versions = "*" files = [ @@ -2349,7 +2264,6 @@ files = [ name = "more-itertools" version = "8.14.0" description = "More routines for operating on iterables, beyond itertools" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -2361,7 +2275,6 @@ files = [ name = "moto" version = "4.2.14" description = "" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -2406,7 +2319,6 @@ xray = ["aws-xray-sdk (>=0.93,!=0.96)", "setuptools"] name = "msgpack" version = "1.0.8" description = "MessagePack serializer" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2472,7 +2384,6 @@ files = [ name = "multidict" version = "6.0.5" description = "multidict implementation" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2572,7 +2483,6 @@ files = [ name = "mypy" version = "1.5.0" description = "Optional static typing for Python" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2614,7 +2524,6 @@ reports = ["lxml"] name = "mypy-extensions" version = "1.0.0" description = "Type system extensions for programs checked with the mypy type checker." -category = "dev" optional = false python-versions = ">=3.5" files = [ @@ -2626,7 +2535,6 @@ files = [ name = "nanoid" version = "2.0.0" description = "A tiny, secure, URL-friendly, unique string ID generator for Python" -category = "main" optional = false python-versions = "*" files = [ @@ -2638,7 +2546,6 @@ files = [ name = "networkx" version = "2.8.8" description = "Python package for creating and manipulating graphs and networks" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2657,7 +2564,6 @@ test = ["codecov (>=2.1)", "pytest (>=7.2)", "pytest-cov (>=4.0)"] name = "newrelic" version = "8.10.0" description = "New Relic Python Agent" -category = "main" optional = false python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*" files = [ @@ -2685,7 +2591,6 @@ infinite-tracing = ["grpcio", "protobuf"] name = "nodeenv" version = "1.9.1" description = "Node.js virtual environment builder" -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" files = [ @@ -2697,7 +2602,6 @@ files = [ name = "notifications-python-client" version = "6.4.1" description = "Python API client for GOV.UK Notify." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -2711,9 +2615,8 @@ requests = ">=2.0.0" [[package]] name = "notifications-utils" -version = "52.3.3" +version = "52.3.5" description = "Shared python code for Notification - Provides logging utils etc." -category = "main" optional = false python-versions = "~3.10.9" files = [] @@ -2748,14 +2651,13 @@ werkzeug = "3.0.4" [package.source] type = "git" url = "https://github.com/cds-snc/notifier-utils.git" -reference = "52.3.3" -resolved_reference = "e8721832bc75903769c6b1b71112fb5eabb5f1ac" +reference = "52.3.5" +resolved_reference = "953ee170b4c47465bef047f1060d17a7702edeeb" [[package]] name = "ordered-set" version = "4.1.0" description = "An OrderedSet is a custom MutableSet that remembers its order, so that every" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2770,7 +2672,6 @@ dev = ["black", "mypy", "pytest"] name = "packaging" version = "24.1" description = "Core utilities for Python packages" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2782,7 +2683,6 @@ files = [ name = "pathspec" version = "0.12.1" description = "Utility library for gitignore style pattern matching of file paths." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2794,7 +2694,6 @@ files = [ name = "phonenumbers" version = "8.13.44" description = "Python version of Google's common library for parsing, formatting, storing and validating international phone numbers." -category = "main" optional = false python-versions = "*" files = [ @@ -2806,7 +2705,6 @@ files = [ name = "platformdirs" version = "4.2.2" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a `user data dir`." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -2823,7 +2721,6 @@ type = ["mypy (>=1.8)"] name = "pluggy" version = "1.5.0" description = "plugin and hook calling mechanisms for python" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -2839,7 +2736,6 @@ testing = ["pytest", "pytest-benchmark"] name = "pre-commit" version = "3.8.0" description = "A framework for managing and maintaining multi-language pre-commit hooks." -category = "main" optional = false python-versions = ">=3.9" files = [ @@ -2858,7 +2754,6 @@ virtualenv = ">=20.10.0" name = "prompt-toolkit" version = "3.0.47" description = "Library for building powerful interactive command lines in Python" -category = "main" optional = false python-versions = ">=3.7.0" files = [ @@ -2873,7 +2768,6 @@ wcwidth = "*" name = "psutil" version = "6.0.0" description = "Cross-platform lib for process and system monitoring in Python." -category = "dev" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" files = [ @@ -2903,7 +2797,6 @@ test = ["enum34", "ipaddress", "mock", "pywin32", "wmi"] name = "psycopg2-binary" version = "2.9.9" description = "psycopg2 - Python-PostgreSQL Database Adapter" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -2985,7 +2878,6 @@ files = [ name = "pwnedpasswords" version = "2.0.0" description = "A Python wrapper for Troy Hunt's Pwned Passwords API." -category = "main" optional = false python-versions = "*" files = [ @@ -2999,7 +2891,6 @@ future = "*" name = "py" version = "1.11.0" description = "library with cross-python path, ini-parsing, io, code, log facilities" -category = "dev" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -3011,7 +2902,6 @@ files = [ name = "py-w3c" version = "0.3.1" description = "W3C services for python." -category = "main" optional = false python-versions = "*" files = [ @@ -3022,7 +2912,6 @@ files = [ name = "pyasn1" version = "0.6.0" description = "Pure-Python implementation of ASN.1 types and DER/BER/CER codecs (X.208)" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3034,7 +2923,6 @@ files = [ name = "pycodestyle" version = "2.11.1" description = "Python style guide checker" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3046,7 +2934,6 @@ files = [ name = "pycparser" version = "2.22" description = "C parser in Python" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3058,7 +2945,6 @@ files = [ name = "pycurl" version = "7.45.3" description = "PycURL -- A Python Interface To The cURL library" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -3104,7 +2990,6 @@ files = [ name = "pydantic" version = "2.8.2" description = "Data validation using Python type hints" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3124,7 +3009,6 @@ email = ["email-validator (>=2.0.0)"] name = "pydantic-core" version = "2.20.1" description = "Core functionality for Pydantic validation and serialization" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3226,7 +3110,6 @@ typing-extensions = ">=4.6.0,<4.7.0 || >4.7.0" name = "pyflakes" version = "3.1.0" description = "passive checker of Python programs" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3238,7 +3121,6 @@ files = [ name = "pygments" version = "2.18.0" description = "Pygments is a syntax highlighting package written in Python." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3253,7 +3135,6 @@ windows-terminal = ["colorama (>=0.4.6)"] name = "pyjwt" version = "2.9.0" description = "JSON Web Token implementation in Python" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3274,7 +3155,6 @@ tests = ["coverage[toml] (==5.0.4)", "pytest (>=6.0.0,<7.0.0)"] name = "pypdf2" version = "1.28.6" description = "A pure-python PDF library capable of splitting, merging, cropping, and transforming PDF files" -category = "main" optional = false python-versions = ">=2.7" files = [ @@ -3286,7 +3166,6 @@ files = [ name = "pyrsistent" version = "0.20.0" description = "Persistent/Functional/Immutable data structures" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3328,7 +3207,6 @@ files = [ name = "pytest" version = "7.4.4" description = "pytest: simple powerful testing with Python" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -3351,7 +3229,6 @@ testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "no name = "pytest-cov" version = "3.0.0" description = "Pytest plugin for measuring coverage." -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -3370,7 +3247,6 @@ testing = ["fields", "hunter", "process-tests", "pytest-xdist", "six", "virtuale name = "pytest-env" version = "0.8.2" description = "py.test plugin that allows you to add environment variables." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -3388,7 +3264,6 @@ test = ["coverage (>=7.2.7)", "pytest-mock (>=3.10)"] name = "pytest-forked" version = "1.6.0" description = "run tests in isolated forked subprocesses" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -3404,7 +3279,6 @@ pytest = ">=3.10" name = "pytest-mock" version = "3.14.0" description = "Thin-wrapper around the mock package for easier use with pytest" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3422,7 +3296,6 @@ dev = ["pre-commit", "pytest-asyncio", "tox"] name = "pytest-mock-resources" version = "2.12.0" description = "A pytest plugin for easily instantiating reproducible mock resources." -category = "dev" optional = false python-versions = "<4,>=3.7" files = [ @@ -3453,7 +3326,6 @@ redshift = ["boto3", "filelock", "moto", "python-on-whales (>=0.22.0)", "sqlpars name = "pytest-xdist" version = "2.5.0" description = "pytest xdist plugin for distributed testing and loop-on-failing modes" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -3475,7 +3347,6 @@ testing = ["filelock"] name = "python-dateutil" version = "2.9.0.post0" description = "Extensions to the standard Python datetime module" -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" files = [ @@ -3490,7 +3361,6 @@ six = ">=1.5" name = "python-dotenv" version = "1.0.1" description = "Read key-value pairs from a .env file and set them as environment variables" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3505,7 +3375,6 @@ cli = ["click (>=5.0)"] name = "python-json-logger" version = "2.0.7" description = "A python library adding a json log formatter" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -3517,7 +3386,6 @@ files = [ name = "python-magic" version = "0.4.27" description = "File type identification using libmagic" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" files = [ @@ -3529,7 +3397,6 @@ files = [ name = "python-on-whales" version = "0.72.0" description = "A Docker client for Python, designed to be fun and intuitive!" -category = "dev" optional = false python-versions = "<4,>=3.8" files = [ @@ -3538,7 +3405,7 @@ files = [ ] [package.dependencies] -pydantic = ">=1.9,<2.0.0 || >=2.1.0,<3" +pydantic = ">=1.9,<2.0.dev0 || >=2.1.dev0,<3" requests = "*" tqdm = "*" typer = ">=0.4.1" @@ -3551,7 +3418,6 @@ test = ["pytest"] name = "pytz" version = "2021.3" description = "World timezone definitions, modern and historical" -category = "main" optional = false python-versions = "*" files = [ @@ -3563,7 +3429,6 @@ files = [ name = "pywin32" version = "306" description = "Python for Window Extensions" -category = "dev" optional = false python-versions = "*" files = [ @@ -3587,7 +3452,6 @@ files = [ name = "pyyaml" version = "6.0.1" description = "YAML parser and emitter for Python" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -3648,7 +3512,6 @@ files = [ name = "pyzmq" version = "26.2.0" description = "Python bindings for 0MQ" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -3770,7 +3633,6 @@ cffi = {version = "*", markers = "implementation_name == \"pypy\""} name = "redis" version = "5.0.8" description = "Python client for Redis database and key-value store" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -3789,7 +3651,6 @@ ocsp = ["cryptography (>=36.0.1)", "pyopenssl (==20.0.1)", "requests (>=2.26.0)" name = "requests" version = "2.31.0" description = "Python HTTP for Humans." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -3811,7 +3672,6 @@ use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] name = "requests-file" version = "2.1.0" description = "File transport adapter for Requests" -category = "main" optional = false python-versions = "*" files = [ @@ -3826,7 +3686,6 @@ requests = ">=1.0.0" name = "requests-mock" version = "1.12.1" description = "Mock out responses from the requests package" -category = "dev" optional = false python-versions = ">=3.5" files = [ @@ -3844,7 +3703,6 @@ fixture = ["fixtures"] name = "requests-toolbelt" version = "1.0.0" description = "A utility belt for advanced users of python-requests" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" files = [ @@ -3859,7 +3717,6 @@ requests = ">=2.0.1,<3.0.0" name = "responses" version = "0.25.3" description = "A utility library for mocking out the `requests` Python library." -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -3879,7 +3736,6 @@ tests = ["coverage (>=6.0.0)", "flake8", "mypy", "pytest (>=7.0.0)", "pytest-asy name = "rfc3987" version = "1.3.8" description = "Parsing and validation of URIs (RFC 3986) and IRIs (RFC 3987)" -category = "dev" optional = false python-versions = "*" files = [ @@ -3891,7 +3747,6 @@ files = [ name = "rich" version = "13.8.0" description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" -category = "dev" optional = false python-versions = ">=3.7.0" files = [ @@ -3910,7 +3765,6 @@ jupyter = ["ipywidgets (>=7.5.1,<9)"] name = "rsa" version = "4.7.2" description = "Pure-Python RSA implementation" -category = "main" optional = false python-versions = ">=3.5, <4" files = [ @@ -3925,7 +3779,6 @@ pyasn1 = ">=0.1.3" name = "s3transfer" version = "0.10.2" description = "An Amazon S3 Transfer Manager" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3943,7 +3796,6 @@ crt = ["botocore[crt] (>=1.33.2,<2.0a.0)"] name = "setuptools" version = "74.0.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -3958,13 +3810,12 @@ cover = ["pytest-cov"] doc = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "pyproject-hooks (!=1.1)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier", "towncrier (<24.7)"] enabler = ["pytest-enabler (>=2.2)"] test = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "ini2toml[lite] (>=0.14)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "jaraco.test", "packaging (>=23.2)", "pip (>=19.1)", "pyproject-hooks (!=1.1)", "pytest (>=6,!=8.1.*)", "pytest-home (>=0.5)", "pytest-perf", "pytest-subprocess", "pytest-timeout", "pytest-xdist (>=3)", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel (>=0.44.0)"] -type = ["importlib-metadata (>=7.0.2)", "jaraco.develop (>=7.21)", "mypy (>=1.11.0,<1.12.0)", "pytest-mypy"] +type = ["importlib-metadata (>=7.0.2)", "jaraco.develop (>=7.21)", "mypy (==1.11.*)", "pytest-mypy"] [[package]] name = "shellingham" version = "1.5.4" description = "Tool to Detect Surrounding Shell" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -3976,7 +3827,6 @@ files = [ name = "simple-salesforce" version = "1.12.6" description = "A basic Salesforce.com REST API client." -category = "main" optional = false python-versions = "*" files = [ @@ -3995,7 +3845,6 @@ zeep = "*" name = "six" version = "1.16.0" description = "Python 2 and 3 compatibility utilities" -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" files = [ @@ -4007,7 +3856,6 @@ files = [ name = "smartypants" version = "2.0.1" description = "Python with the SmartyPants" -category = "main" optional = false python-versions = "*" files = [ @@ -4018,7 +3866,6 @@ files = [ name = "sqlalchemy" version = "1.4.52" description = "Database Abstraction Library" -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" files = [ @@ -4098,7 +3945,6 @@ sqlcipher = ["sqlcipher3_binary"] name = "sqlalchemy-stubs" version = "0.4" description = "SQLAlchemy stubs and mypy plugin" -category = "dev" optional = false python-versions = "*" files = [ @@ -4114,7 +3960,6 @@ typing-extensions = ">=3.7.4" name = "sqlalchemy2-stubs" version = "0.0.2a38" description = "Typing Stubs for SQLAlchemy 1.4" -category = "dev" optional = false python-versions = ">=3.6" files = [ @@ -4129,7 +3974,6 @@ typing-extensions = ">=3.7.4" name = "statsd" version = "3.3.0" description = "A simple statsd client." -category = "main" optional = false python-versions = "*" files = [ @@ -4141,7 +3985,6 @@ files = [ name = "strict-rfc3339" version = "0.7" description = "Strict, simple, lightweight RFC3339 functions" -category = "dev" optional = false python-versions = "*" files = [ @@ -4152,7 +3995,6 @@ files = [ name = "tldextract" version = "3.5.0" description = "Accurately separates a URL's subdomain, domain, and public suffix, using the Public Suffix List (PSL). By default, this includes the public ICANN TLDs and their exceptions. You can optionally support the Public Suffix List's private domains as well." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -4170,7 +4012,6 @@ requests-file = ">=1.4" name = "tomli" version = "2.0.1" description = "A lil' TOML parser" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -4182,7 +4023,6 @@ files = [ name = "tqdm" version = "4.66.5" description = "Fast, Extensible Progress Meter" -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -4203,7 +4043,6 @@ telegram = ["requests"] name = "typer" version = "0.12.5" description = "Typer, build great CLIs. Easy to code. Based on Python type hints." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -4221,7 +4060,6 @@ typing-extensions = ">=3.7.4.3" name = "types-boto" version = "2.49.18.9" description = "Typing stubs for boto" -category = "dev" optional = false python-versions = "*" files = [ @@ -4233,7 +4071,6 @@ files = [ name = "types-cffi" version = "1.16.0.20240331" description = "Typing stubs for cffi" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4248,7 +4085,6 @@ types-setuptools = "*" name = "types-mock" version = "4.0.15.2" description = "Typing stubs for mock" -category = "dev" optional = false python-versions = "*" files = [ @@ -4260,7 +4096,6 @@ files = [ name = "types-pyopenssl" version = "24.1.0.20240722" description = "Typing stubs for pyOpenSSL" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4276,7 +4111,6 @@ types-cffi = "*" name = "types-python-dateutil" version = "2.9.0.20240316" description = "Typing stubs for python-dateutil" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4288,7 +4122,6 @@ files = [ name = "types-pytz" version = "2022.7.1.2" description = "Typing stubs for pytz" -category = "dev" optional = false python-versions = "*" files = [ @@ -4300,7 +4133,6 @@ files = [ name = "types-redis" version = "4.6.0.20240425" description = "Typing stubs for redis" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4316,7 +4148,6 @@ types-pyOpenSSL = "*" name = "types-requests" version = "2.32.0.20240712" description = "Typing stubs for requests" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4331,7 +4162,6 @@ urllib3 = ">=2" name = "types-setuptools" version = "73.0.0.20240822" description = "Typing stubs for setuptools" -category = "dev" optional = false python-versions = ">=3.8" files = [ @@ -4343,7 +4173,6 @@ files = [ name = "typing-extensions" version = "4.12.2" description = "Backported and Experimental Type Hints for Python 3.8+" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -4355,7 +4184,6 @@ files = [ name = "tzdata" version = "2024.1" description = "Provider of IANA time zone data" -category = "main" optional = false python-versions = ">=2" files = [ @@ -4367,7 +4195,6 @@ files = [ name = "unidecode" version = "1.3.8" description = "ASCII transliterations of Unicode text" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -4379,7 +4206,6 @@ files = [ name = "urllib3" version = "2.2.2" description = "HTTP library with thread-safe connection pooling, file post, and more." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -4397,7 +4223,6 @@ zstd = ["zstandard (>=0.18.0)"] name = "vine" version = "5.1.0" description = "Python promises." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -4409,7 +4234,6 @@ files = [ name = "virtualenv" version = "20.26.3" description = "Virtual Python Environment builder" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -4430,7 +4254,6 @@ test = ["covdefaults (>=2.3)", "coverage (>=7.2.7)", "coverage-enable-subprocess name = "wcwidth" version = "0.2.13" description = "Measures the displayed width of unicode strings in a terminal" -category = "main" optional = false python-versions = "*" files = [ @@ -4442,7 +4265,6 @@ files = [ name = "webencodings" version = "0.5.1" description = "Character encoding aliases for legacy web content" -category = "main" optional = false python-versions = "*" files = [ @@ -4454,7 +4276,6 @@ files = [ name = "werkzeug" version = "3.0.4" description = "The comprehensive WSGI web application library." -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -4472,7 +4293,6 @@ watchdog = ["watchdog (>=2.3)"] name = "wrapt" version = "1.16.0" description = "Module for decorators, wrappers and monkey patching." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -4552,7 +4372,6 @@ files = [ name = "xmltodict" version = "0.13.0" description = "Makes working with XML feel like you are working with JSON" -category = "dev" optional = false python-versions = ">=3.4" files = [ @@ -4564,7 +4383,6 @@ files = [ name = "yarl" version = "1.9.4" description = "Yet another URL library" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -4668,7 +4486,6 @@ multidict = ">=4.0" name = "zeep" version = "4.2.1" description = "A Python SOAP client" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -4696,7 +4513,6 @@ xmlsec = ["xmlsec (>=0.6.1)"] name = "zope-event" version = "5.0" description = "Very basic event publishing system" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -4715,7 +4531,6 @@ test = ["zope.testrunner"] name = "zope-interface" version = "7.0.3" description = "Interfaces for Python" -category = "main" optional = false python-versions = ">=3.8" files = [ @@ -4766,4 +4581,4 @@ testing = ["coverage (>=5.0.3)", "zope.event", "zope.testing"] [metadata] lock-version = "2.0" python-versions = "~3.10.9" -content-hash = "a57921f03421f99d89e5449aee7fc8b2dd3bb05b2163cd0b0b16ce3a9f7ef2bc" +content-hash = "61bcdd1269b2ad2d719e75d1ccbfcc2a2072efca7db178b6d9892b6cbac30714" diff --git a/pyproject.toml b/pyproject.toml index a8723c3e81..ae58c89454 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -65,7 +65,7 @@ Werkzeug = "3.0.4" MarkupSafe = "2.1.5" # REVIEW: v2 is using sha512 instead of sha1 by default (in v1) itsdangerous = "2.2.0" -notifications-utils = { git = "https://github.com/cds-snc/notifier-utils.git", tag = "52.3.3" } +notifications-utils = { git = "https://github.com/cds-snc/notifier-utils.git", tag = "52.3.5" } # rsa = "4.9 # awscli 1.22.38 depends on rsa<4.8 typing-extensions = "4.12.2" diff --git a/tests/app/db.py b/tests/app/db.py index 53217bffe0..066701a431 100644 --- a/tests/app/db.py +++ b/tests/app/db.py @@ -195,6 +195,7 @@ def create_template( template_category=None, postage=None, process_type="normal", + text_direction_rtl=False, ): if not template_category: data = { @@ -220,6 +221,7 @@ def create_template( "folder": folder, "template_category": template_category, "process_type": process_type, + "text_direction_rtl": text_direction_rtl, } if template_type == LETTER_TYPE: data["postage"] = postage or "second" diff --git a/tests/app/template/test_rest.py b/tests/app/template/test_rest.py index 0e0bc9a4fe..bb30b48fa4 100644 --- a/tests/app/template/test_rest.py +++ b/tests/app/template/test_rest.py @@ -1824,3 +1824,85 @@ def test_create_template_with_category_ff_on( assert response["data"]["process_type_column"] == process_type assert response["data"]["process_type"] == calculated_process_type assert response["data"]["template_category_id"] == str(tc.id) + + +@pytest.mark.parametrize( + "original_text_direction, updated_text_direction, expected_original, expected_after_update", + [ + (None, True, False, True), + (True, False, True, False), + ], +) +def test_template_updated_when_rtl_changes( + admin_request, + sample_user, + notify_api, + sample_template_category, + original_text_direction, + updated_text_direction, + expected_original, + expected_after_update, +): + service = create_service(service_name="service_1") + template_data = { + "service": service, + "template_type": "email", + "template_name": "testing template", + "subject": "Template subject", + "content": "Dear Sir/Madam, Hello. Yours Truly, The Government.", + "template_category": sample_template_category, + "process_type": "normal", + "text_direction_rtl": original_text_direction, + } + template = create_template(**template_data) + + assert template.text_direction_rtl is expected_original + + # change the RTL property + updated_template_data = { + "service": str(service.id), + "template_category": str(sample_template_category.id), + "template_type": template_data["template_type"], + "template_name": template_data["template_name"], + "subject": template_data["subject"], + "content": template_data["content"], + "process_type": template_data["process_type"], + } + + if updated_text_direction is not None: + updated_template_data["text_direction_rtl"] = updated_text_direction + + response = admin_request.post( + "template.update_template", + service_id=service.id, + template_id=template.id, + _data=updated_template_data, + _expected_status=200, + ) + + assert response["data"]["text_direction_rtl"] == expected_after_update + + +@pytest.mark.parametrize("text_direction, expected_text_direction", [(True, True), (False, False), (None, False)]) +def test_template_can_be_created_with_text_direction( + admin_request, sample_user, sample_template_category, text_direction, expected_text_direction +): + service = create_service(service_name="service_1") + + template_data = { + "name": "new name", + "template_type": "email", + "content": "some content here :)", + "subject": "yo", + "service": str(service.id), + "created_by": str(sample_user.id), + "template_category_id": str(sample_template_category.id), + "process_type": "normal", + } + + if text_direction is not None: + template_data["text_direction_rtl"] = text_direction + + response = admin_request.post("template.create_template", service_id=service.id, _data=template_data, _expected_status=201) + + assert response["data"]["text_direction_rtl"] == expected_text_direction