Dark magics about variable names in python
Change Log | API | Playground
pip install -U varname
-
Core features:
- Retrieving names of variables a function/class call is assigned to from inside it, using
varname
. - Retrieving variable names directly, using
nameof
- Detecting next immediate attribute name, using
will
- Fetching argument names/sources passed to a function using
argname2
(argname
is superseded byargname2
)
- Retrieving names of variables a function/class call is assigned to from inside it, using
-
Other helper APIs (built based on core features):
- A value wrapper to store the variable name that a value is assigned to, using
Wrapper
- A decorator to register
__varname__
to functions/classes, usingregister
- A
debug
function to print variables with their names and values
- A value wrapper to store the variable name that a value is assigned to, using
Thanks goes to these awesome people/projects:
@alexmojaki |
executing |
Special thanks to @HanyuuLu to give up the name varname
in pypi for this project.
-
From inside a function
from varname import varname def function(): return varname() func = function() # func == 'func'
When there are intermediate frames:
def wrapped(): return function() def function(): # retrieve the variable name at the 2nd frame from this one return varname(frame=2) func = wrapped() # func == 'func'
Or use
ignore
to ignore the wrapped frame:def wrapped(): return function() def function(): return varname(ignore=wrapped) func = wrapped() # func == 'func'
Calls from standard libraries are ignored by default:
import asyncio async def function(): return varname() func = asyncio.run(function()) # func == 'func'
-
Retrieving name of a class instance
class Foo: def __init__(self): self.id = varname() def copy(self): # also able to fetch inside a method call copied = Foo() # copied.id == 'copied' copied.id = varname() # assign id to whatever variable name return copied foo = Foo() # foo.id == 'foo' foo2 = foo.copy() # foo2.id == 'foo2'
-
Multiple variables on Left-hand side
# since v0.5.4 def func(): return varname(multi_vars=True) a = func() # a == ('a', ) a, b = func() # (a, b) == ('a', 'b') [a, b] = func() # (a, b) == ('a', 'b') # hierarchy is also possible a, (b, c) = func() # (a, b, c) == ('a', 'b', 'c')
-
Some unusual use
def function(): return varname() func = [function()] # func == ['func'] func = [function(), function()] # func == ['func', 'func'] func = function(), function() # func = ('func', 'func') func = func1 = function() # func == func1 == 'func' # a warning will be shown # since you may not want func1 to be 'func' x = func(y = func()) # x == 'x' # get part of the name func_abc = function()[-3:] # func_abc == 'abc' # function alias supported now function2 = function func = function2() # func == 'func' a = lambda: 0 a.b = function() # a.b == 'b' # Since v0.1.3 # We can ask varname to raise exceptions # if it fails to detect the variable name def get_name(raise_exc): return varname(raise_exc=raise_exc) a = {} a['b'] = get_name(True) # VarnameRetrievingError a['b'] = get_name(False) # None
-
Registering
__varname__
to functionsfrom varname.helpers import register @register def function(): return __varname__ func = function() # func == 'func'
# arguments also allowed (frame, ignore and raise_exc) @register(frame=2) def function(): return __varname__ def wrapped(): return function() func = wrapped() # func == 'func'
-
Registering
__varname__
as a class property@register class Foo: ... foo = Foo() # foo.__varname__ == 'foo'
from varname import varname, nameof
a = 1
nameof(a) # 'a'
b = 2
nameof(a, b) # ('a', 'b')
def func():
return varname() + '_suffix'
f = func() # f == 'f_suffix'
nameof(f) # 'f'
# get full names of (chained) attribute calls
func.a = func
nameof(func.a, vars_only=False) # 'func.a'
func.a.b = 1
nameof(func.a.b, vars_only=False) # 'func.a.b'
from varname import will
class AwesomeClass:
def __init__(self):
self.will = None
def permit(self):
self.will = will(raise_exc=False)
if self.will == 'do':
# let self handle do
return self
raise AttributeError('Should do something with AwesomeClass object')
def do(self):
if self.will != 'do':
raise AttributeError("You don't have permission to do")
return 'I am doing!'
awesome = AwesomeClass()
awesome.do() # AttributeError: You don't have permission to do
awesome.permit() # AttributeError: Should do something with AwesomeClass object
awesome.permit().do() == 'I am doing!'
from varname import argname2
def func(a, b=1):
print(argname2('a'))
x = y = z = 2
func(x) # prints: x
def func2(a, b=1):
print(argname2('a', 'b'))
func2(y, b=x) # prints: ('y', 'x')
# allow expressions
def func3(a, b=1):
print(argname2('a', 'b', vars_only=False))
func3(x+y, y+x) # prints: ('x+y', 'y+x')
# positional and keyword arguments
def func4(*args, **kwargs):
print(argname2('args[1]', 'kwargs["c"]'))
func4(y, x, c=z) # prints: ('x', 'z')
from varname.helpers import Wrapper
foo = Wrapper(True)
# foo.name == 'foo'
# foo.value == True
bar = Wrapper(False)
# bar.name == 'bar'
# bar.value == False
def values_to_dict(*args):
return {val.name: val.value for val in args}
mydict = values_to_dict(foo, bar)
# {'foo': True, 'bar': False}
from varname.helpers import debug
a = 'value'
b = object()
debug(a) # DEBUG: a='value'
debug(b) # DEBUG: b=<object object at 0x2b70580e5f20>
debug(a, b)
# DEBUG: a='value'
# DEBUG: b=<object object at 0x2b70580e5f20>
debug(a, b, merge=True)
# DEBUG: a='value', b=<object object at 0x2b70580e5f20>
debug(a, repr=False, prefix='') # a=value
# also debug an expression
debug(a+a) # DEBUG: a+a='valuevalue'
# If you want to disable it:
debug(a+a, vars_only=True) # error
varname
is all depending on executing
package to look for the node.
The node executing
detects is ensured to be the correct one (see this).
It partially works with environments where other AST magics apply, including
pytest
, ipython
, macropy
, birdseye
, reticulate
with R
, etc. Neither
executing
nor varname
is 100% working with those environments. Use
it at your own risk.
For example:
- This will not work with
pytest
:a = 1 assert nameof(a) == 'a' # pytest manipulated the ast here # do this instead name_a = nameof(a) assert name_a == 'a'