Skip to content

Commit 5b2d79e

Browse files
committed
Complete the aliases management
1 parent 1cb877f commit 5b2d79e

File tree

3 files changed

+76
-74
lines changed

3 files changed

+76
-74
lines changed

python/aliases.py

+8-4
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,14 @@ def check(root):
133133
global ok
134134
ok = True
135135
managed = collect_managed(root)
136+
for filepath, content in managed.items():
137+
if not filepath.exists():
138+
print(f"Error: {filepath} should exist.")
139+
ok = False
140+
continue
141+
if filepath.read_text() != content:
142+
print(f"Error: {filepath} has wrong content.")
143+
ok = False
136144
ignored = [root / path for path in SOURCES + IGNORED]
137145

138146
def check_file(path):
@@ -142,10 +150,6 @@ def check_file(path):
142150
if path not in managed:
143151
print(f"Error: {path} shouldn't exist.")
144152
ok = False
145-
return
146-
if path.read_text() != managed[path]:
147-
print(f"Error: {path} has wrong content.")
148-
ok = False
149153

150154
traverse(root, check_file)
151155

python/hopsworks/internal/aliases.py

+68-45
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,12 @@
2727
from dataclasses import dataclass, field
2828
from typing import Dict, Optional
2929

30-
from hopsworks.internal.exceptions import InternalError
30+
31+
class InternalAliasError(Exception):
32+
"""Internal hopsworks exception related to aliases.
33+
34+
Ideally, this exception should never happen, as it means misconfiguration of aliases, for example, if a public alias is requested for a method of a class.
35+
"""
3136

3237

3338
@dataclass
@@ -53,15 +58,17 @@ def update(self, other: Alias.InModule):
5358
self.deprecated |= other.deprecated
5459
if self.available_until:
5560
if self.available_until != other.available_until:
56-
raise InternalError(
61+
raise InternalAliasError(
5762
"Deprecated alias is declared available until different releases."
5863
)
5964
else:
6065
self.available_until = other.available_until
6166

6267
def __post_init__(self):
6368
if "." in self.import_name:
64-
raise InternalError("Impossible to create alias for not importable symbol.")
69+
raise InternalAliasError(
70+
"Impossible to create alias for not importable symbol."
71+
)
6572

6673
def add(self, *in_modules: InModule):
6774
for im in in_modules:
@@ -80,98 +87,112 @@ def __new__(cls):
8087
return Registry
8188

8289
aliases: Dict[str, Alias] = {}
90+
modules = {}
8391

8492
@staticmethod
8593
def get_modules():
86-
modules = {}
94+
for module, exclude, paths in Registry.modules.values():
95+
for name in set(x for x, _ in inspect.getmembers(module)) - exclude:
96+
alias = Alias(module.__name__, name)
97+
alias.add(*(Alias.InModule(p) for p in paths))
98+
Registry.add(alias)
99+
res = {}
87100
for alias in Registry.aliases.values():
88101
for im in alias.in_modules.values():
89102
from_import = alias.from_module, alias.import_name, im
90103
parts = im.in_module.split(".")
91104
for i in range(1, len(parts)):
92-
modules.setdefault(".".join(parts[:i]), [])
93-
modules.setdefault(im.in_module, []).append(from_import)
94-
return modules
105+
res.setdefault(".".join(parts[:i]), [])
106+
res.setdefault(im.in_module, []).append(from_import)
107+
return res
95108

96109
@staticmethod
97110
def add(*aliases):
98111
for alias in aliases:
99112
Registry.aliases.setdefault(alias.get_id(), alias).update(alias)
100113

114+
@staticmethod
115+
def add_module(module, exclude, paths):
116+
Registry.modules[module.__name__] = module, exclude, paths
117+
101118

102-
def public(*paths: str, as_alias: Optional[str] = None):
119+
def public(
120+
*paths: str,
121+
as_alias: Optional[str] = None,
122+
deprecated: bool = False,
123+
available_until: Optional[str] = None,
124+
):
103125
"""Make a function or class publically available.
104126
105127
If you want to publish a constant, use `publish`.
106-
Note that it is impossible to create an alias for a variable, i.e., it is impossible to make a change of a variable in one module to propogate to another variable in another module.
107128
108129
# Arguments
109130
paths: the import paths under which the entity is publically avilable; effectively results in generation of aliases in all of the paths for the entity.
110131
as_alias: make the alias of the specified name.
132+
deprecated: make the alias deprected; use of the entity outside hopsworks will print a warning, saying that it is going to be removed from the public API in one of the future releases. See `deprecated` decorator for the implementation of construction of the deprecated objects.
133+
available_until: the first hopsworks release in which the entity will become unavailable, defaults to `None`; if the release is known, it is reoprted to the external user in the warning and `deprected` becomes set up.
111134
"""
112135

136+
if available_until:
137+
deprecated = True
138+
113139
def decorator(symbol):
114140
if not hasattr(symbol, "__qualname__"):
115-
raise InternalError("The symbol should be importable to be public.")
141+
raise InternalAliasError("The symbol should be importable to be public.")
116142
alias = Alias(symbol.__module__, symbol.__qualname__)
117-
alias.add(*(Alias.InModule(p, as_alias) for p in paths))
143+
alias.add(
144+
*(Alias.InModule(p, as_alias, deprecated, available_until) for p in paths)
145+
)
118146
Registry.add(alias)
119147
return symbol
120148

121149
return decorator
122150

123151

124-
class Publisher:
125-
"""Publish all of the names defined inside this context.
152+
def publish(*paths: str):
153+
"""Publish all of the names defined in this module after this call.
126154
127155
Since `public` decorator works only for classes and functions, this context should be used for public constants.
128156
It is also useful for bulk publishing.
129157
Note that it is impossible to create an alias for a variable, i.e., it is impossible to make a change of a variable in one module to propogate to another variable in another module.
130158
159+
In case you need to publish something from the begining of a module, use `Publisher`.
160+
161+
If you need to deprecate an alias, use `public` instead.
162+
131163
# Arguments
132164
paths: the import paths under which the names declared in this context will be publically available; effectively results in generation of aliases in all of the paths for all the names declared in the context.
133165
"""
134166

135-
def __init__(self, *paths: str):
136-
self.paths = set(paths)
167+
caller = inspect.getmodule(inspect.stack()[1][0])
168+
exclude = set(x for x, _ in inspect.getmembers(caller))
169+
Registry.add_module(caller, exclude, paths)
137170

138-
def __enter__(self):
139-
caller = inspect.getmodule(inspect.stack()[1][0])
140-
self.exclude = set(x for x, _ in inspect.getmembers(caller))
141-
142-
def __exit__(self, _exc_type, _exc_value, _traceback):
143-
caller = inspect.getmodule(inspect.stack()[1][0])
144-
for name in set(x for x, _ in inspect.getmembers(caller)) - self.exclude:
145-
alias = Alias(caller.__name__, name)
146-
alias.add(*(Alias.InModule(p) for p in self.paths))
147-
Registry.add(alias)
148171

172+
class Publisher:
173+
"""Publish all of the names defined inside this context.
149174
150-
def deprecated_public(
151-
*paths: str,
152-
available_until: Optional[str] = None,
153-
as_alias: Optional[str] = None,
154-
):
155-
"""Make public aliases as a deprecated versions of a class or a function.
175+
This class is intended for bulk publishing of entitities which are to be declared in the begining of a module, so that publish is not usable; in other cases, use publish instead.
176+
Note that it is impossible to create an alias for a variable, i.e., it is impossible to make a change of a variable in one module to propogate to another variable in another module.
156177
157-
Use of the entity outside hopsworks will print a warning, saying that it is going to be removed from the public API in one of the future releases.
158-
See `deprecated` decorator for the implementation of construction of the deprecated objects.
178+
If you need to deprecate an alias, use `public` instead.
159179
160180
# Arguments
161-
paths: the import paths under which the entity is publically avilable; effectively results in generation of deprecated aliases in all of the paths for the entity.
162-
available_until: the first hopsworks release in which the entity will become unavailable, defaults to `None`; if the release is known, it is reoprted to the external user in the warning.
163-
as_alias: make the alias of the specified name.
181+
paths: the import paths under which the names declared in this context will be publically available; effectively results in generation of aliases in all of the paths for all the names declared in the context.
164182
"""
165183

166-
def decorator(symbol):
167-
if not hasattr(symbol, "__qualname__"):
168-
raise InternalError("The symbol should be importable to be public.")
169-
alias = Alias(symbol.__module__, symbol.__qualname__)
170-
alias.add(*(Alias.InModule(p, as_alias, True, available_until) for p in paths))
171-
Registry.add(alias)
172-
return symbol
184+
def __init__(self, *paths: str):
185+
self.paths = set(paths)
186+
self.caller = inspect.getmodule(inspect.stack()[1][0])
173187

174-
return decorator
188+
def __enter__(self):
189+
self.exclude = set(x for x, _ in inspect.getmembers(self.caller))
190+
191+
def __exit__(self, _exc_type, _exc_value, _traceback):
192+
for name in set(x for x, _ in inspect.getmembers(self.caller)) - self.exclude:
193+
alias = Alias(self.caller.__name__, name)
194+
alias.add(*(Alias.InModule(p) for p in self.paths))
195+
Registry.add(alias)
175196

176197

177198
class DeprecatedCallWarning(Warning):
@@ -212,6 +233,8 @@ def deprecated_f(*args, **kwargs):
212233

213234
return deprecated_f
214235
else:
215-
raise InternalError("Deprecation of something else than class or function.")
236+
raise InternalAliasError(
237+
"Deprecation of something else than class or function."
238+
)
216239

217240
return deprecate

python/hopsworks/internal/exceptions.py

-25
This file was deleted.

0 commit comments

Comments
 (0)