From a225a8b41fd9cdf022de26cf9076b22be9d27a56 Mon Sep 17 00:00:00 2001 From: Quinn Batten Date: Fri, 11 Nov 2022 14:33:16 -0500 Subject: [PATCH] feat(streammaps): Add datetime function to simpleeval env in streammaps --- docs/stream_maps.md | 2 ++ singer_sdk/mapper.py | 2 ++ tests/core/test_mapper.py | 11 +++++++++++ 3 files changed, 15 insertions(+) diff --git a/docs/stream_maps.md b/docs/stream_maps.md index bf44046cd..1526f63af 100644 --- a/docs/stream_maps.md +++ b/docs/stream_maps.md @@ -160,6 +160,8 @@ can be referenced directly by mapping expressions. of the hash's hex digest. - This is defined by the SDK internally with native python: `hashlib.md5(.encode("utf-8")).hexdigest()`. +- `datetime` - This is the datetime module object from the Python standard library. You can access + datetime.datetime, datetime.timedelta, etc. #### Built-in Variable Names diff --git a/singer_sdk/mapper.py b/singer_sdk/mapper.py index 48918def2..2fd6108ca 100644 --- a/singer_sdk/mapper.py +++ b/singer_sdk/mapper.py @@ -7,6 +7,7 @@ import abc import copy +import datetime import hashlib import logging from typing import Any, Callable @@ -292,6 +293,7 @@ def functions(self) -> dict[str, Callable]: """ funcs: dict[str, Any] = simpleeval.DEFAULT_FUNCTIONS.copy() funcs["md5"] = md5 + funcs["datetime"] = datetime return funcs def _eval( diff --git a/tests/core/test_mapper.py b/tests/core/test_mapper.py index ba9e0f8d8..1ef0cc792 100644 --- a/tests/core/test_mapper.py +++ b/tests/core/test_mapper.py @@ -36,6 +36,7 @@ def sample_catalog_dict() -> dict: Property("name", StringType), Property("owner_email", StringType), Property("description", StringType), + Property("description", StringType), ).to_dict() foobars_schema = PropertiesList( Property("the", StringType), @@ -70,21 +71,25 @@ def sample_stream(): "name": "tap-something", "owner_email": "sample1@example.com", "description": "Comment A", + "create_date": "2019-01-01", }, { "name": "my-tap-something", "owner_email": "sample2@example.com", "description": "Comment B", + "create_date": "2020-01-01", }, { "name": "target-something", "owner_email": "sample3@example.com", "description": "Comment C", + "create_date": "2021-01-01", }, { "name": "not-atap", "owner_email": "sample4@example.com", "description": "Comment D", + "create_date": "2022-01-01", }, ], "foobars": [ @@ -107,6 +112,7 @@ def transform_stream_maps(): "email_hash": "md5(config['hash_seed'] + owner_email)", "description": "'[masked]'", "description2": "str('[masked]')", + "create_year": "int(datetime.date.fromisoformat(create_date).year)", "int_test": "int('0')", "__else__": None, }, @@ -125,6 +131,7 @@ def transformed_result(stream_map_config): ), "description": "[masked]", "description2": "[masked]", + "create_year": 2019, "int_test": 0, }, { @@ -135,6 +142,7 @@ def transformed_result(stream_map_config): ), "description": "[masked]", "description2": "[masked]", + "create_year": 2020, "int_test": 0, }, { @@ -145,6 +153,7 @@ def transformed_result(stream_map_config): ), "description": "[masked]", "description2": "[masked]", + "create_year": 2021, "int_test": 0, }, { @@ -155,6 +164,7 @@ def transformed_result(stream_map_config): ), "description": "[masked]", "description2": "[masked]", + "create_year": 2022, "int_test": 0, }, ], @@ -174,6 +184,7 @@ def transformed_schemas(): Property("email_hash", StringType), Property("description", StringType), Property("description2", StringType), + Property("create_year", IntegerType), Property("int_test", IntegerType), ).to_dict(), "foobars": PropertiesList(