Python is easy, just use the contextmanager decorator to turn the generator into a context manager that is also at the same time a decorator factory! For real though i love this❤
@ArcRCG4 ай бұрын
This is cool, this decorator-context manager equivalence reflects that a decorator what basically is doing is "managing" the "context" of the wrapped function and that a context manager is basically is "decorating" an annonymous IIFE. One really beautiful abstraction.
@sadhlife6 ай бұрын
wow that's a hack that I can use a lot more of
@fabiolean2 ай бұрын
I had no idea that creating a context manager like that also created a decorator. Blowing my mind, here.
@mrswats6 ай бұрын
Excuse me, what the hell. This is super useful! Plus, much easier to test! Holy crap, I will try to remember to use this. Was this a planned feature? Or a coincidence by how Python is built? Or added afterwards...?
@anthonywritescode6 ай бұрын
yeah it got added after contextmanager was introduced
@Finezzato6 ай бұрын
@@anthonywritescode so you basically dismissed decorators and then you used decorators
@anthonywritescode6 ай бұрын
???
@neonwatty6 ай бұрын
good god man this is fantastic - thanks!
@FunkyELF6 ай бұрын
Cool, didn't know you could do this. This can only be done when the decorator returns the unmodified output of the given function being called with unmodified input... correct? Clearly these kinds of decorators exist... like ones that register stuff, log stuff, etc... but a lot of decorators are used to modify either input, output, coerce stuff, etc. If this can't help in those cases the beginning of this video is actually a great reference for how to properly type regular decorators.
@NoProblem762 ай бұрын
There’s also the contextdecorator class that’s worth looking into
@bryanbanda6 ай бұрын
So obvious and straightforward! Love it
@ericwadebrown6 ай бұрын
This is why I was sad to see verbose typing in Python. Reminds of the same thing that happened in Java. Of course, it's still optional, but still.
@helleye3116 ай бұрын
I don't even write python that much, randomly saw this vid in recommended. It looks amazing! Even without the fact that you don't need to type anything, it's so much nicer than defining 3 nested functions. Tbh python decorators are the only feature i miss in other languages, but they're so nice.
@spyr0th3dr4g0n6 ай бұрын
That's such a nice and smart simplification, completely decoupling the function typing rather than going generic with it. Can the context manager decorator do something like logging the first parameter while also timing it, or are the parameters/return values completely out of reach?
@float326 ай бұрын
You can get access to anything with the inspect module, but things will start getting slow.
@anthonywritescode6 ай бұрын
nope. though arguably those are need special handling anyway and I would not recommend personally
@TheTugaMachine6 ай бұрын
Damn wish I knew this sooner. However the drawback of not accessing arguments and return values kinda limit the scope of this approach (e.g. custom caching decorator)
@senseikoudai61866 ай бұрын
This is incredible!
@theendlessriver136 ай бұрын
How long has it been like that? Holy - soooo much easier. Is there any downside to this? Why would you ever use the "old" way? Imagine not using functools either...
@anthonywritescode6 ай бұрын
the only downside is no argument hackery signature modification but imo that's a good thing
@KhalilMuhammad6 ай бұрын
Amazing! So all these times I've been creating context managers, I could've also used them as a decorator? 🤯
@trupalcanada6 ай бұрын
This feels like black magic
@insanecbrotha10 күн бұрын
hm, interesting, but you cannot access the decorated function object or the passed args/kwargs, right? So this is only for side-effect decorations, I think.
@bacon4life6 ай бұрын
That's super smart
@zeyadmoustafakamal6 ай бұрын
Wait how you made nano like this ?? I would like to look at your dotfiles or something like this as I can't believe that nano can be like this. It's like vim or neovim but it's nicer.
@JohnZakaria6 ай бұрын
This isn't nano. This is babi. The gigachad here wrote it himself
@anthonywritescode6 ай бұрын
framing this lol. this is my editor I made github.com/asottile/babi
@zeyadmoustafakamal6 ай бұрын
@@anthonywritescode Wow what a crazy one. I will look at it :)
@447xpro4 ай бұрын
Thanks!
@shunitsu__6 ай бұрын
Very interesting! Is this possible for concatenate introduced in PEP 612? like the one example in your previous video adding a logger
@anthonywritescode6 ай бұрын
this sort of intentionally prevents you from doing nasty signature mutation. the concatenate example in the other video was to demonstrate the feature and not really something I would ever do in reality
@Xx00aquaman00xX6 ай бұрын
This is cool but what if you want override/pass the args/kwargs of the func or you want to get the name of the func?
@anthonywritescode6 ай бұрын
you can't, but I see that as a good thing
@MichalPlichta6 ай бұрын
That's kind if new.... Contex manager working as decorator 🧐.... I need try....
@notmymainaccount4776 ай бұрын
2:40 is there a specific reason why you use time.monotonic instead of time.perf_counter, because as far as i can remember you have always used time.monotonic in your videos?
@anthonywritescode6 ай бұрын
it's an alias on the platforms I care about and good enough on windows
@notmymainaccount4776 ай бұрын
@@anthonywritescode ah i see, thanks
@SQADIqw6 ай бұрын
Does it make any sense if you are not into typing?
@anthonywritescode6 ай бұрын
yes, it is still a simpler way to make decorators
@yorailevi67476 ай бұрын
Is there no down sides to this?
@flipbit036 ай бұрын
This is awesome, and makes for a much nicer typing experience. Quick question on the @timing_ctx example: is "try: finally:" really needed in this case, or just a "nice to have" for the sake of completeness? i.e. would the section after the "yield" be skipped on an exception inside the decorated function if we ditch the try/finally? I'm asking this because imho, the decorator "setup/teardown" sections would look even better/simpler/more readable without the extra indentation that the finally: block unfortunately imposes on the "teardown" section of it. Thanks in advance, asottile, you are a legend!
@anthonywritescode6 ай бұрын
yeah finally is necessary to work in the context of exceptions -- this is also true in the decorator code too (if I skipped it there it's a mistake!)
@murphygreen84846 ай бұрын
What is the [None, None, None] typing for on the generator?
@philipreinhold11806 ай бұрын
Generator has a yield type (the type in the yield), send type (usually only required for coroutines) and return type (the final returned value after the generator is exhausted)
@murphygreen84846 ай бұрын
@@philipreinhold1180 thank you!
@anthonywritescode6 ай бұрын
kzbin.info/www/bejne/eoXImJl5g6aLpa8 for my video on the subject
@1rbroderi6 ай бұрын
Anyway to generalize this to create a decorator that decorates another decorator?
@anthonywritescode6 ай бұрын
I mean, it should work
@wexwexexort6 ай бұрын
I just wondered if you use an IDE at work
@anthonywritescode6 ай бұрын
why wouldn't I use my own text editor?
@thagreatone4026 ай бұрын
Blasphemy
@float326 ай бұрын
Am I noticing correctly that the context manager method has 1000x overhead? 2ms vs the 0.002ms of the decorator!?
@hemerythrin6 ай бұрын
It's 1.7us for the decorator and 1.9us for the context manager. The 200ms one is after he added the with block containing a sleep.
@anthonywritescode6 ай бұрын
I intentionally added a sleep so the timing output was interesting
@float326 ай бұрын
Ahh, I was just looking at the time beyond the sleep (like 0.207 at 6:19). Maybe it’s a sleep resolution thing.
@anthonywritescode6 ай бұрын
ah probably just python show plus vm plus recording
@wagneralberto54566 ай бұрын
first!
@NeatMemesDotCom6 ай бұрын
At this you should probably go writing rust code. Easier than all this