123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387 |
- import json
- import re
- import shutil
- import tempfile
- from pathlib import Path
- from unittest.mock import Mock, mock_open
- import pytest
- from typer.testing import CliRunner
- from reflex import constants
- from reflex.config import Config
- from reflex.reflex import cli
- from reflex.testing import chdir
- from reflex.utils.prerequisites import (
- CpuInfo,
- _update_next_config,
- cached_procedure,
- get_cpu_info,
- initialize_requirements_txt,
- rename_imports_and_app_name,
- )
- runner = CliRunner()
- @pytest.mark.parametrize(
- "config, export, expected_output",
- [
- (
- Config(
- app_name="test",
- ),
- False,
- 'module.exports = {basePath: "", compress: true, trailingSlash: true, staticPageGenerationTimeout: 60};',
- ),
- (
- Config(
- app_name="test",
- static_page_generation_timeout=30,
- ),
- False,
- 'module.exports = {basePath: "", compress: true, trailingSlash: true, staticPageGenerationTimeout: 30};',
- ),
- (
- Config(
- app_name="test",
- next_compression=False,
- ),
- False,
- 'module.exports = {basePath: "", compress: false, trailingSlash: true, staticPageGenerationTimeout: 60};',
- ),
- (
- Config(
- app_name="test",
- frontend_path="/test",
- ),
- False,
- 'module.exports = {basePath: "/test", compress: true, trailingSlash: true, staticPageGenerationTimeout: 60};',
- ),
- (
- Config(
- app_name="test",
- frontend_path="/test",
- next_compression=False,
- ),
- False,
- 'module.exports = {basePath: "/test", compress: false, trailingSlash: true, staticPageGenerationTimeout: 60};',
- ),
- (
- Config(
- app_name="test",
- ),
- True,
- 'module.exports = {basePath: "", compress: true, trailingSlash: true, staticPageGenerationTimeout: 60, output: "export", distDir: "_static"};',
- ),
- ],
- )
- def test_update_next_config(config, export, expected_output):
- output = _update_next_config(config, export=export)
- assert output == expected_output
- @pytest.mark.parametrize(
- ("transpile_packages", "expected_transpile_packages"),
- (
- (
- ["foo", "@bar/baz"],
- ["@bar/baz", "foo"],
- ),
- (
- ["foo", "@bar/baz", "foo", "@bar/baz@3.2.1"],
- ["@bar/baz", "foo"],
- ),
- ),
- )
- def test_transpile_packages(transpile_packages, expected_transpile_packages):
- output = _update_next_config(
- Config(app_name="test"),
- transpile_packages=transpile_packages,
- )
- transpile_packages_match = re.search(r"transpilePackages: (\[.*?\])", output)
- transpile_packages_json = transpile_packages_match.group(1) # pyright: ignore [reportOptionalMemberAccess]
- actual_transpile_packages = sorted(json.loads(transpile_packages_json))
- assert actual_transpile_packages == expected_transpile_packages
- def test_initialize_requirements_txt_no_op(mocker):
- # File exists, reflex is included, do nothing
- mocker.patch("pathlib.Path.exists", return_value=True)
- mocker.patch(
- "charset_normalizer.from_path",
- return_value=Mock(best=lambda: Mock(encoding="utf-8")),
- )
- mock_fp_touch = mocker.patch("pathlib.Path.touch")
- open_mock = mock_open(read_data="reflex==0.6.7")
- mocker.patch("pathlib.Path.open", open_mock)
- initialize_requirements_txt()
- assert open_mock.call_count == 1
- assert open_mock.call_args.kwargs["encoding"] == "utf-8"
- assert open_mock().write.call_count == 0
- mock_fp_touch.assert_not_called()
- def test_initialize_requirements_txt_missing_reflex(mocker):
- # File exists, reflex is not included, add reflex
- mocker.patch("pathlib.Path.exists", return_value=True)
- mocker.patch(
- "charset_normalizer.from_path",
- return_value=Mock(best=lambda: Mock(encoding="utf-8")),
- )
- open_mock = mock_open(read_data="random-package=1.2.3")
- mocker.patch("pathlib.Path.open", open_mock)
- initialize_requirements_txt()
- # Currently open for read, then open for append
- assert open_mock.call_count == 2
- for call_args in open_mock.call_args_list:
- assert call_args.kwargs["encoding"] == "utf-8"
- assert (
- open_mock().write.call_args[0][0]
- == f"\n{constants.RequirementsTxt.DEFAULTS_STUB}{constants.Reflex.VERSION}\n"
- )
- def test_initialize_requirements_txt_not_exist(mocker):
- # File does not exist, create file with reflex
- mocker.patch("pathlib.Path.exists", return_value=False)
- open_mock = mock_open()
- mocker.patch("pathlib.Path.open", open_mock)
- initialize_requirements_txt()
- assert open_mock.call_count == 2
- # By default, use utf-8 encoding
- for call_args in open_mock.call_args_list:
- assert call_args.kwargs["encoding"] == "utf-8"
- assert open_mock().write.call_count == 1
- assert (
- open_mock().write.call_args[0][0]
- == f"{constants.RequirementsTxt.DEFAULTS_STUB}{constants.Reflex.VERSION}\n"
- )
- def test_requirements_txt_cannot_detect_encoding(mocker):
- mocker.patch("pathlib.Path.exists", return_value=True)
- mock_open = mocker.patch("builtins.open")
- mocker.patch(
- "charset_normalizer.from_path",
- return_value=Mock(best=lambda: None),
- )
- initialize_requirements_txt()
- mock_open.assert_not_called()
- def test_requirements_txt_other_encoding(mocker):
- mocker.patch("pathlib.Path.exists", return_value=True)
- mocker.patch(
- "charset_normalizer.from_path",
- return_value=Mock(best=lambda: Mock(encoding="utf-16")),
- )
- initialize_requirements_txt()
- open_mock = mock_open(read_data="random-package=1.2.3")
- mocker.patch("pathlib.Path.open", open_mock)
- initialize_requirements_txt()
- # Currently open for read, then open for append
- assert open_mock.call_count == 2
- for call_args in open_mock.call_args_list:
- assert call_args.kwargs["encoding"] == "utf-16"
- assert (
- open_mock().write.call_args[0][0]
- == f"\n{constants.RequirementsTxt.DEFAULTS_STUB}{constants.Reflex.VERSION}\n"
- )
- def test_cached_procedure():
- call_count = 0
- @cached_procedure(tempfile.mktemp(), payload_fn=lambda: "constant")
- def _function_with_no_args():
- nonlocal call_count
- call_count += 1
- _function_with_no_args()
- assert call_count == 1
- _function_with_no_args()
- assert call_count == 1
- call_count = 0
- @cached_procedure(
- tempfile.mktemp(),
- payload_fn=lambda *args, **kwargs: f"{repr(args), repr(kwargs)}",
- )
- def _function_with_some_args(*args, **kwargs):
- nonlocal call_count
- call_count += 1
- _function_with_some_args(1, y=2)
- assert call_count == 1
- _function_with_some_args(1, y=2)
- assert call_count == 1
- _function_with_some_args(100, y=300)
- assert call_count == 2
- _function_with_some_args(100, y=300)
- assert call_count == 2
- def test_get_cpu_info():
- cpu_info = get_cpu_info()
- assert cpu_info is not None
- assert isinstance(cpu_info, CpuInfo)
- assert cpu_info.model_name is not None
- for attr in ("manufacturer_id", "model_name", "address_width"):
- value = getattr(cpu_info, attr)
- assert value.strip() if attr != "address_width" else value
- @pytest.fixture
- def temp_directory():
- temp_dir = tempfile.mkdtemp()
- yield Path(temp_dir)
- shutil.rmtree(temp_dir)
- @pytest.mark.parametrize(
- "config_code,expected",
- [
- ("rx.Config(app_name='old_name')", 'rx.Config(app_name="new_name")'),
- ('rx.Config(app_name="old_name")', 'rx.Config(app_name="new_name")'),
- ("rx.Config('old_name')", 'rx.Config("new_name")'),
- ('rx.Config("old_name")', 'rx.Config("new_name")'),
- ],
- )
- def test_rename_imports_and_app_name(temp_directory, config_code, expected):
- file_path = temp_directory / "rxconfig.py"
- content = f"""
- config = {config_code}
- """
- file_path.write_text(content)
- rename_imports_and_app_name(file_path, "old_name", "new_name")
- updated_content = file_path.read_text()
- expected_content = f"""
- config = {expected}
- """
- assert updated_content == expected_content
- def test_regex_edge_cases(temp_directory):
- file_path = temp_directory / "example.py"
- content = """
- from old_name.module import something
- import old_name
- from old_name import something_else as alias
- from old_name
- """
- file_path.write_text(content)
- rename_imports_and_app_name(file_path, "old_name", "new_name")
- updated_content = file_path.read_text()
- expected_content = """
- from new_name.module import something
- import new_name
- from new_name import something_else as alias
- from new_name
- """
- assert updated_content == expected_content
- def test_cli_rename_command(temp_directory):
- foo_dir = temp_directory / "foo"
- foo_dir.mkdir()
- (foo_dir / "__init__").touch()
- (foo_dir / ".web").mkdir()
- (foo_dir / "assets").mkdir()
- (foo_dir / "foo").mkdir()
- (foo_dir / "foo" / "__init__.py").touch()
- (foo_dir / "rxconfig.py").touch()
- (foo_dir / "rxconfig.py").write_text(
- """
- import reflex as rx
- config = rx.Config(
- app_name="foo",
- )
- """
- )
- (foo_dir / "foo" / "components").mkdir()
- (foo_dir / "foo" / "components" / "__init__.py").touch()
- (foo_dir / "foo" / "components" / "base.py").touch()
- (foo_dir / "foo" / "components" / "views.py").touch()
- (foo_dir / "foo" / "components" / "base.py").write_text(
- """
- import reflex as rx
- from foo.components import views
- from foo.components.views import *
- from .base import *
- def random_component():
- return rx.fragment()
- """
- )
- (foo_dir / "foo" / "foo.py").touch()
- (foo_dir / "foo" / "foo.py").write_text(
- """
- import reflex as rx
- import foo.components.base
- from foo.components.base import random_component
- class State(rx.State):
- pass
- def index():
- return rx.text("Hello, World!")
- app = rx.App()
- app.add_page(index)
- """
- )
- with chdir(temp_directory / "foo"):
- result = runner.invoke(cli, ["rename", "bar"])
- assert result.exit_code == 0
- assert (foo_dir / "rxconfig.py").read_text() == (
- """
- import reflex as rx
- config = rx.Config(
- app_name="bar",
- )
- """
- )
- assert (foo_dir / "bar").exists()
- assert not (foo_dir / "foo").exists()
- assert (foo_dir / "bar" / "components" / "base.py").read_text() == (
- """
- import reflex as rx
- from bar.components import views
- from bar.components.views import *
- from .base import *
- def random_component():
- return rx.fragment()
- """
- )
- assert (foo_dir / "bar" / "bar.py").exists()
- assert not (foo_dir / "bar" / "foo.py").exists()
- assert (foo_dir / "bar" / "bar.py").read_text() == (
- """
- import reflex as rx
- import bar.components.base
- from bar.components.base import random_component
- class State(rx.State):
- pass
- def index():
- return rx.text("Hello, World!")
- app = rx.App()
- app.add_page(index)
- """
- )
|