Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix[lang]: stateless modules should not be initialized #4347

Open
wants to merge 32 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
1b1ac45
forbid initialization of a stateless module
sandbubbles Nov 3, 2024
3a9e29d
fix tests so they do not initialize a stateless module
sandbubbles Nov 3, 2024
5527244
lint
sandbubbles Nov 3, 2024
3c695da
set immutables as stateless
sandbubbles Nov 4, 2024
5000efc
add tests
sandbubbles Nov 4, 2024
e3d7582
fix stateless test
sandbubbles Nov 4, 2024
c92e1ad
make immutables state
sandbubbles Nov 4, 2024
cf91784
fix test
sandbubbles Nov 4, 2024
1ab6433
test transient in cancun
sandbubbles Nov 5, 2024
28c9be7
make init function as state
sandbubbles Nov 5, 2024
0c5f645
remove one wrong test
sandbubbles Nov 5, 2024
47efc8e
refactor is_stateless into ModuleT method
sandbubbles Nov 9, 2024
5fe6c84
simplify is_stateless
charles-cooper Nov 20, 2024
fececb7
promote some properties to cached properties
charles-cooper Nov 20, 2024
4cc01bb
add comment to explain a test
sandbubbles Dec 5, 2024
ba19783
rename function to is_initializable
sandbubbles Dec 5, 2024
b668f41
remove redundant storage variables
sandbubbles Dec 5, 2024
6c7a30f
make modules with nonreentrant initializable
sandbubbles Dec 5, 2024
948d047
test that nonreentrant function use state
sandbubbles Dec 5, 2024
3dbbeb6
update comment
sandbubbles Dec 5, 2024
1f4a2d6
remove uses decl from initializable
sandbubbles Dec 6, 2024
be5aa0b
rename the function so the boolean matches
sandbubbles Dec 6, 2024
7b2f38b
refactor comment
sandbubbles Dec 7, 2024
bb4944f
change is_not_initializable to is_initializable
sandbubbles Dec 12, 2024
b41bc28
rename variable
sandbubbles Dec 15, 2024
2bcba89
remove unnecessary phonies and initializes
sandbubbles Dec 15, 2024
7f2f452
clean up tests
sandbubbles Dec 15, 2024
f64109f
remove phony variables to fix tests with "uses"
sandbubbles Dec 30, 2024
bf7307c
modify test to expect a successful compilation
sandbubbles Dec 30, 2024
a075979
mark uses declaration initializable
sandbubbles Dec 30, 2024
b67ab3f
Merge branch 'master' into fix/stateless-modules-can-be-initialized
sandbubbles Dec 30, 2024
ef0c49d
add comments to tests
sandbubbles Dec 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions tests/functional/codegen/features/test_transient.py
Original file line number Diff line number Diff line change
Expand Up @@ -570,3 +570,15 @@ def foo() -> (uint256[3], uint256, uint256, uint256):

c = get_contract(main, input_bundle=input_bundle)
assert c.foo() == ([1, 2, 3], 1, 2, 42)


def test_transient_is_state(make_input_bundle):
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's please add a comment here that we're testing the initialization and how it relates to state

lib = """
message: transient(bool)
"""
main = """
import lib
initializes: lib
"""
input_bundle = make_input_bundle({"lib.vy": lib, "main.vy": main})
compile_code(main, input_bundle=input_bundle)
4 changes: 4 additions & 0 deletions tests/functional/codegen/modules/test_nonreentrant.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
def test_export_nonreentrant(make_input_bundle, get_contract, tx_failed):
lib1 = """
phony: uint32
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this feels off, see my comment at is_stateless


interface Foo:
def foo() -> uint256: nonpayable

Expand Down Expand Up @@ -38,6 +40,8 @@ def __default__():

def test_internal_nonreentrant(make_input_bundle, get_contract, tx_failed):
lib1 = """
phony: uint32

interface Foo:
def foo() -> uint256: nonpayable

Expand Down
133 changes: 133 additions & 0 deletions tests/functional/syntax/modules/test_initializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -835,6 +835,8 @@ def test_uses_skip_import(make_input_bundle):
lib2 = """
import lib1

phony: uint32

@internal
def foo():
pass
Expand Down Expand Up @@ -1418,3 +1420,134 @@ def set_some_mod():
compile_code(main, input_bundle=input_bundle)
assert e.value._message == "module `lib3.vy` is used but never initialized!"
assert e.value._hint is None


storage_var_modules = [
"""
phony: uint32
""",
"""
ended: public(bool)
""",
]


@pytest.mark.parametrize("module", storage_var_modules)
def test_initializes_on_modules_with_state_related_vars(module, make_input_bundle):
main = """
import lib
initializes: lib
"""
input_bundle = make_input_bundle({"lib.vy": module, "main.vy": main})
compile_code(main, input_bundle=input_bundle)


def test_initializes_on_modules_with_immutables(make_input_bundle):
lib = """
foo: immutable(int128)

@deploy
def __init__():
foo = 2
"""

main = """
import lib
initializes: lib

@deploy
def __init__():
lib.__init__()
"""
input_bundle = make_input_bundle({"lib.vy": lib, "main.vy": main})
compile_code(main, input_bundle=input_bundle)


stateless_modules = [
"""
""",
"""
@internal
@pure
def foo(x: uint256, y: uint256) -> uint256:
return unsafe_add(x & y, (x ^ y) >> 1)
""",
"""
FOO: constant(int128) = 128
""",
]


@pytest.mark.parametrize("module", stateless_modules)
def test_forbids_initializes_on_stateless_modules(module, make_input_bundle):
main = """
import lib
initializes: lib
"""
input_bundle = make_input_bundle({"lib.vy": module, "main.vy": main})
with pytest.raises(StructureException):
compile_code(main, input_bundle=input_bundle)


def test_initializes_on_modules_with_uses(make_input_bundle):
lib0 = """
import lib1
uses: lib1

@external
def foo() -> uint32:
return lib1.phony
"""
lib1 = """
phony: uint32
"""
main = """
import lib1
initializes: lib1

import lib0
initializes: lib0[lib1 := lib1]
"""
input_bundle = make_input_bundle({"lib1.vy": lib1, "lib0.vy": lib0, "main.vy": main})
compile_code(main, input_bundle=input_bundle)


def test_initializes_on_modules_with_initializes(make_input_bundle):
lib0 = """
import lib1
initializes: lib1
"""
lib1 = """
phony: uint32
"""
main = """
import lib0
initializes: lib0
"""
input_bundle = make_input_bundle({"lib1.vy": lib1, "lib0.vy": lib0, "main.vy": main})
compile_code(main, input_bundle=input_bundle)


def test_initializes_on_modules_with_init_function(make_input_bundle):
lib = """
interface Foo:
def foo(): payable

@deploy
def __init__():
extcall Foo(self).foo()
"""
main = """
import lib
initializes: lib

@deploy
def __init__():
lib.__init__()

@external
def foo():
pass
"""
input_bundle = make_input_bundle({"lib.vy": lib, "main.vy": main})
compile_code(main, input_bundle=input_bundle)
8 changes: 8 additions & 0 deletions tests/unit/compiler/test_abi.py
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,8 @@ def bar(x: {type}):

def test_exports_abi(make_input_bundle):
lib1 = """
phony: uint32

@external
def foo():
pass
Expand Down Expand Up @@ -330,6 +332,8 @@ def __init__():
def test_event_export_from_init(make_input_bundle):
# test that events get exported when used in init functions
lib1 = """
phony: uint32

event MyEvent:
pass

Expand Down Expand Up @@ -361,6 +365,8 @@ def __init__():
def test_event_export_from_function_export(make_input_bundle):
# test events used in exported functions are exported
lib1 = """
phony: uint32
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe we shouldn't add phony here -- the code change is telling us to remove initializes: lib1


event MyEvent:
pass

Expand Down Expand Up @@ -396,6 +402,8 @@ def foo():
def test_event_export_unused_function(make_input_bundle):
# test events in unused functions are not exported
lib1 = """
phony: uint32

event MyEvent:
pass

Expand Down
5 changes: 5 additions & 0 deletions vyper/semantics/analysis/module.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@
check_modifiability,
get_exact_type_from_node,
get_expr_info,
is_stateless,
)
from vyper.semantics.data_locations import DataLocation
from vyper.semantics.namespace import Namespace, get_namespace, override_global_namespace
Expand Down Expand Up @@ -409,6 +410,10 @@ def visit_InitializesDecl(self, node):
module_info = get_expr_info(module_ref).module_info
if module_info is None:
raise StructureException("Not a module!", module_ref)
if is_stateless(module_info.module_node):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe should we also have this restriction on UsesDecl -- @cyberthirst wdyt?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sorry, missed the ping

i think that makes sense - uses indicates that we're accessing the used module's state - and thus the module should actually have state

raise StructureException(
f"Cannot initialize a stateless module {module_info.alias}!", module_ref
)

used_modules = {i.module_t: i for i in module_info.module_t.used_modules}

Expand Down
16 changes: 16 additions & 0 deletions vyper/semantics/analysis/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -736,3 +736,19 @@ def validate_kwargs(node: vy_ast.Call, members: dict[str, VyperType], typeclass:
msg = f"{typeclass} instantiation missing fields:"
msg += f" {', '.join(list(missing))}"
raise InstantiationException(msg, node)


def is_stateless(module: vy_ast.Module):
"""
Determine whether a module is stateless by examining its top-level declarations.
A module has state if it contains storage variables, transient variables, or
immutables, or if it includes a "uses" or "initializes" declaration.
"""
for i in module.body:
if isinstance(i, (vy_ast.InitializesDecl, vy_ast.UsesDecl)):
return False
if isinstance(i, vy_ast.VariableDecl) and not i.is_constant:
return False
if isinstance(i, vy_ast.FunctionDef) and i.name == "__init__":
return False
return True
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i think this belongs better on ModuleT

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay, moved it.

Loading