Created
September 27, 2020 22:08
-
-
Save expobrain/96913312d882af6db612a93cebee5dec to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from typing import cast | |
import libcst as cst | |
import libcst.matchers as m | |
from libcst.codemod import VisitorBasedCodemodCommand | |
from libcst.codemod.visitors import AddImportsVisitor, RemoveImportsVisitor | |
class DatetimeUtcnow(VisitorBasedCodemodCommand): | |
DESCRIPTION: str = "Converts from datetime.utcnow() to datetime.utc()" | |
timezone_utc_matcher = m.Arg( | |
value=m.Attribute( | |
value=m.Name(value="timezone"), attr=m.Name(value="utc") | |
), | |
keyword=m.Name(value="tzinfo"), | |
) | |
utc_matcher = m.Arg( | |
value=m.OneOf( | |
m.Name(value="utc"), | |
m.Name(value="UTC"), | |
m.Attribute(value=m.Name(value="pytz",), attr=m.Name(value="UTC")), | |
), | |
keyword=m.Name(value="tzinfo"), | |
) | |
datetime_utcnow_matcher = m.Call( | |
func=m.Attribute( | |
value=m.Name(value="datetime"), attr=m.Name(value="utcnow") | |
), | |
args=[], | |
) | |
datetime_datetime_utcnow_matcher = m.Call( | |
func=m.Attribute( | |
value=m.Attribute( | |
value=m.Name(value="datetime"), attr=m.Name(value="datetime") | |
), | |
attr=m.Name(value="utcnow"), | |
), | |
args=[], | |
) | |
datetime_replace_matcher = m.Call( | |
func=m.Attribute( | |
value=datetime_utcnow_matcher, attr=m.Name(value="replace") | |
), | |
args=[m.OneOf(timezone_utc_matcher, utc_matcher)], | |
) | |
datetime_datetime_replace_matcher = m.Call( | |
func=m.Attribute( | |
value=datetime_datetime_utcnow_matcher, | |
attr=m.Name(value="replace"), | |
), | |
args=[m.OneOf(timezone_utc_matcher, utc_matcher)], | |
) | |
timedelta_replace_matcher = m.Call( | |
func=m.Attribute( | |
value=m.BinaryOperation( | |
left=m.OneOf( | |
datetime_utcnow_matcher, datetime_datetime_utcnow_matcher | |
), | |
operator=m.Add(), | |
), | |
attr=m.Name(value="replace"), | |
), | |
args=[m.OneOf(timezone_utc_matcher, utc_matcher)], | |
) | |
utc_localize_matcher = m.Call( | |
func=m.Attribute( | |
value=m.Name(value="UTC"), attr=m.Name(value="localize"), | |
), | |
args=[ | |
m.Arg( | |
value=m.OneOf( | |
datetime_utcnow_matcher, datetime_datetime_utcnow_matcher | |
) | |
) | |
], | |
) | |
def _update_imports(self): | |
RemoveImportsVisitor.remove_unused_import(self.context, "pytz") | |
RemoveImportsVisitor.remove_unused_import(self.context, "pytz", "utc") | |
RemoveImportsVisitor.remove_unused_import(self.context, "pytz", "UTC") | |
RemoveImportsVisitor.remove_unused_import( | |
self.context, "datetime", "timezone" | |
) | |
AddImportsVisitor.add_needed_import( | |
self.context, "bulb.platform.common.timezones", "UTC" | |
) | |
@m.leave(datetime_utcnow_matcher) | |
def datetime_utcnow_call( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.Call: | |
self._update_imports() | |
return updated_node.with_changes( | |
func=cst.Attribute( | |
value=cst.Name(value="datetime"), attr=cst.Name("now") | |
), | |
args=[cst.Arg(value=cst.Name(value="UTC"))], | |
) | |
@m.leave(datetime_datetime_utcnow_matcher) | |
def datetime_datetime_utcnow_call( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.Call: | |
self._update_imports() | |
return updated_node.with_changes( | |
func=cst.Attribute( | |
value=cst.Attribute( | |
value=cst.Name(value="datetime"), | |
attr=cst.Name(value="datetime"), | |
), | |
attr=cst.Name(value="now"), | |
), | |
args=[cst.Arg(value=cst.Name(value="UTC"))], | |
) | |
@m.leave(datetime_replace_matcher) | |
def datetime_replace( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.Call: | |
self._update_imports() | |
return updated_node.with_changes( | |
func=cst.Attribute( | |
value=cst.Name(value="datetime"), attr=cst.Name("now") | |
), | |
args=[cst.Arg(value=cst.Name(value="UTC"))], | |
) | |
@m.leave(datetime_datetime_replace_matcher) | |
def datetime_datetime_replace( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.Call: | |
self._update_imports() | |
return updated_node.with_changes( | |
func=cst.Attribute( | |
value=cst.Attribute( | |
value=cst.Name(value="datetime"), | |
attr=cst.Name(value="datetime"), | |
), | |
attr=cst.Name(value="now"), | |
), | |
args=[cst.Arg(value=cst.Name(value="UTC"))], | |
) | |
@m.leave(timedelta_replace_matcher) | |
def timedelta_replace( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.BinaryOperation: | |
self._update_imports() | |
return cast( | |
cst.BinaryOperation, | |
cast(cst.Attribute, cast(cst.Call, updated_node).func).value, | |
) | |
@m.leave(utc_localize_matcher) | |
def utc_localize( | |
self, original_node: cst.Call, updated_node: cst.Call | |
) -> cst.Call: | |
self._update_imports() | |
return cast(cst.Call, updated_node.args[0].value) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import textwrap | |
from libcst.codemod import CodemodTest | |
from codemods.datetime_utcnow import DatetimeUtcnow | |
class DatetimeUtcnowTests(CodemodTest): | |
TRANSFORM = DatetimeUtcnow | |
def test_datetime_call(self) -> None: | |
before = textwrap.dedent( | |
""" | |
datetime.utcnow() | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_datetime_datetime_call(self) -> None: | |
before = textwrap.dedent( | |
""" | |
datetime.datetime.utcnow() | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_datetime_replace_timezone_utc(self) -> None: | |
before = textwrap.dedent( | |
""" | |
from datetime import timezone | |
datetime.utcnow().replace(tzinfo=timezone.utc) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_datetime_replace_utc(self) -> None: | |
before = textwrap.dedent( | |
""" | |
from pytz import utc | |
datetime.utcnow().replace(tzinfo=utc) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_datetime_datetime_replace_utc(self) -> None: | |
before = textwrap.dedent( | |
""" | |
from pytz import utc | |
datetime.datetime.utcnow().replace(tzinfo=utc) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_timedelta_replace_utc(self) -> None: | |
before = textwrap.dedent( | |
""" | |
from pytz import utc | |
(datetime.datetime.utcnow() + timedelta).replace(tzinfo=utc) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
(datetime.datetime.now(UTC) + timedelta) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_timedelta_replace_utc_2(self) -> None: | |
# LibCST cannot understand if UTC from pytz is unused or not | |
before = textwrap.dedent( | |
""" | |
from pytz import UTC | |
datetime.datetime.utcnow().replace(tzinfo=UTC) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from pytz import UTC | |
from bulb.platform.common.timezones import UTC | |
datetime.datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_timedelta_replace_utc_3(self) -> None: | |
before = textwrap.dedent( | |
""" | |
import pytz | |
datetime.datetime.utcnow().replace(tzinfo=pytz.UTC) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_timedelta_replace_timezone_utc(self) -> None: | |
before = textwrap.dedent( | |
""" | |
from datetime import timezone | |
(datetime.utcnow() + timedelta).replace(tzinfo=timezone.utc) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
(datetime.now(UTC) + timedelta) | |
""" | |
) | |
self.assertCodemod(before, after) | |
def test_utc_localize(self) -> None: | |
before = textwrap.dedent( | |
""" | |
UTC.localize(datetime.utcnow()) | |
UTC.localize(datetime.datetime.utcnow()) | |
""" | |
) | |
after = textwrap.dedent( | |
""" | |
from bulb.platform.common.timezones import UTC | |
datetime.now(UTC) | |
datetime.datetime.now(UTC) | |
""" | |
) | |
self.assertCodemod(before, after) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/bin/bash -eu | |
cd "$(dirname "$0")" | |
source_paths=${@:2} | |
python -m libcst.tool codemod datetime_utcnow.DatetimeUtcnow $source_paths |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment