Brought to you by Michael and Brian - take a Talk Python course or get Brian's pytest book


Transcript #195: Runtime type checking for Python type hints

Return to episode page view on github
Recorded on Wednesday, Aug 12, 2020.

00:00 Hello and welcome to Python Bytes, where we deliver Python news and headlines directly to your earbuds.

00:04 This is episode 195, recorded August 12th, 2020.

00:10 I'm Michael Kennedy.

00:11 And I am Brian Okken.

00:12 And this episode is brought to you by all the cool work that we're doing.

00:15 Tell you more about this.

00:16 Right now, I kind of just want to think about paying attention to stuff, Brian.

00:20 Maybe watching things.

00:22 I'll, like, look closer.

00:24 Yes, watch carefully.

00:25 So, actually, this thing sent over by Preyson Daniel, he sends us a bunch of good topics and links.

00:32 And this one's called Watchdog.

00:34 And maybe you've heard of it.

00:34 It's the foundation of some web frameworks, for example, and things like that to know if, say, a file has changed that you're editing.

00:42 Does it need to auto-restart the website?

00:44 So when you refresh it, it actually reruns it.

00:46 Things like that.

00:47 But Watchdog is a cool little library that you can use just on your own to know if something has changed.

00:55 So, basically, it's a simple little API.

00:57 You create this thing called an observer.

00:58 And you tell it to just start watching in some way.

01:03 You give it a path.

01:04 It can recursively look.

01:06 You can give it, like, a pattern or whatnot.

01:07 And it will say, just basically start firing events back on this observer thing when stuff happens.

01:15 File is created.

01:15 File is deleted.

01:16 File is modified.

01:17 And so on.

01:18 Nice.

01:18 This is cool.

01:19 Yeah.

01:19 Isn't that cool?

01:20 It also comes with a CLI script called WatchMeDo.

01:25 And what you can do with WatchMeDo is just type WatchMeDo, all one word, log.

01:31 And wherever your current working directory is, when you type that, it will just start watching

01:36 for files that change right there, just like on the command line.

01:40 Oh, okay.

01:40 Cool.

01:40 So, if you're, you know, you don't want to write a program, but you just happen to be somewhere

01:44 and you're like, what is happening?

01:46 Like, what files are being modified or being touched or being changed here?

01:49 And you can just type that and boom, there it goes.

01:51 That's cool, right?

01:52 That actually is really cool.

01:54 Yeah.

01:54 We have a build process that part of it is mucking up some directories and it'd be kind

01:59 of cool to use something like this.

02:00 Yeah.

02:01 Absolutely.

02:02 And so you could just pip install WatchDog and then just type WatchMeDo space log wherever

02:07 you want it to know.

02:08 There's a bunch of other things it does that you can pass commands and stuff, I suppose.

02:13 Like, you could ask it to recursively watch or whatever, but yeah, I haven't had a use

02:18 case for that.

02:19 So, but yeah, this is a cool recommendation from Preycin.

02:21 Yeah, nice.

02:22 The example that I think I could possibly use it for is, you know, one of the things

02:27 that we do that makes Python Bytes, .fm, as well as all the Talk Python, Talk Python training

02:33 sites ridiculously fast as a user is we don't require you to re-download almost any of the

02:40 static content, anything JavaScript, any image in the entire site, you know, CSS, all of those

02:46 things, they're cached for a year, right?

02:50 And so a problem with caching stuff for a long time is if you redeploy a new version,

02:55 something's wacky, right?

02:58 Something is really weird.

02:58 Like, I have this problem with Twitter and Firefox.

03:01 I'll go there and periodically it'll just say something went wrong, can't load it, and a command

03:06 R, a hard refresh always fixes it because there's probably some JavaScript file that's like out of

03:11 sync with some other part of some API or some random thing like that.

03:15 And so what we do to make sure that never happens is we look at every one of those static

03:20 files that we have a year-long cache on and we read it and we create the hash and we put

03:24 question mark hash equals the hash of the file on the end of it.

03:29 So that, you know, that's a separate file.

03:32 If it ever changes one character, if it changes, the hash changes and it's now a totally separate

03:38 thing in the web browser cache, right?

03:39 All of this is perfect.

03:41 The one drawback is to make things fast, you're not rehashing all the images on every

03:45 request.

03:46 Is it just says, have I seen this file before?

03:49 Do I already compute the hash?

03:51 Just use it, right?

03:52 So that's fine.

03:52 But that means that little hash refresh trick only really works if I restart the website.

03:58 If like, say, only a CSS file changed.

04:01 I could use this watchdog to watch all the static files that we're hashing.

04:05 If any of them change, just instantly recompute the hash.

04:09 That's it.

04:10 So that would allow me to do like push deploys of just, you know, static content changes without

04:16 kicking the website at all.

04:17 And it would just magically like redetect those and start rolling.

04:20 I think I might start using watchdog for that.

04:23 Anyway, that's my use case that makes me excited.

04:25 And I'm glad you pronounced watch me do because I looked at it and I went watch meadow.

04:29 That's a weird name.

04:30 That is a weird.

04:31 I think it's a watch me do.

04:33 Watch me do is better.

04:34 Yeah.

04:34 Yeah.

04:35 You know what else is weird?

04:36 Weird HTTP status codes.

04:39 Like I'm very familiar with 400 bad requests.

04:42 That's when somebody's created an API and I talked to it badly and they've created the API correctly.

04:48 Or I get a 500 where there is their stuff has crashed because I sent them something bad and

04:54 they've written their API badly.

04:56 But somewhere in between there lives some odd thing.

05:00 Right?

05:01 Yes.

05:01 Have you heard of error code 418 before?

05:04 I'm a teapot.

05:07 This is just great.

05:08 So I love it.

05:09 I love it.

05:10 Status code 418.

05:11 I'm a teapot.

05:13 Any attempt to brew a coffee with a teapot should result in an error code 418.

05:18 I'm a teapot.

05:18 The resulting entity may be short and stout.

05:21 So the resulting entity may be short and stout.

05:25 I love it.

05:26 This actually got brought up in a conversation I was having this morning.

05:29 A colleague of mine, Andy Howe, suggested it.

05:32 He said, so when is Python 3.9 going to come out?

05:36 Because hasn't it been a while?

05:37 And as a reminder to everybody, Python 3.9 candidate release or the release candidate one

05:43 or RCCR or whatever is out now.

05:46 So you can play with it.

05:47 It's probably, I've been using it.

05:49 It's safe to use.

05:50 But the schedule for the release, final release is in October, I believe.

05:57 And then after that, bug fixes, releases every month are planned, if there are any.

06:03 Andy said, you know, my favorite enhancement for Python 3.9 is this 418.

06:10 I'm a teapot.

06:11 This is new in Python 3.9.

06:13 HTTP library was missing the status code 418.

06:18 I'm a teapot.

06:19 And now it has it in there.

06:20 But wow, I've got some links for you.

06:23 Because in the show notes, this is referenced from HTCPCP, which is Hypertext Coffee Pot Control Protocol.

06:33 And this came out in 1998 as an April Fool's joke.

06:37 And then it's kind of part of it now.

06:40 So it's part of HTTP, sort of.

06:42 Another fun error code.

06:45 So the document, the entire document for HTCPCP, says that most error codes share the same status codes as HTTP.

06:54 However, 418 is separate.

06:57 But also now HTTP can use that also.

07:00 But also 406, not acceptable.

07:03 So 406 is this response code may be returned if the operator of the coffee pot cannot comply with the accept addition request.

07:12 Unless the request was in ahead.

07:14 So, and it should return the list of possible coffee additions.

07:21 This is bizarre that this is in the world.

07:23 This is awesome.

07:25 I really love it.

07:26 I don't realize, I didn't realize it had been taken so far as this.

07:29 This is great.

07:30 And, you know, if you go and look at the thing you link to, the Hypertext Coffee Pot Control Protocol, HTCPCP slash 1.0.

07:39 The whole memo describes so much of this thing.

07:43 And it even talks about things like crossing firewalls.

07:46 Like, modern coffee pots do not use fire.

07:49 However, a firewall is useful in protecting it.

07:52 It also addresses the security considerations.

07:55 Anyone who gets between me and my morning coffee should be insecure.

07:58 Filtration is not a good virus protection method.

08:12 I mean, it just goes on and on.

08:13 It's beautiful.

08:14 Yes.

08:14 And there's a security layer that I forgot about that's funny, too.

08:21 That's part of this.

08:22 But I can't find it right now.

08:23 Yeah, no worries.

08:24 I threw in a quick link also to this place called HTPstatuses.com, which this is a slightly more serious take, although it does, of course, include the coffee pot.

08:36 But it's just, if you were like, what the heck should I be doing, you know, in this situation?

08:41 Like, here's the error codes here.

08:42 So, and if you just click on them, it gives you a, like, really nice summary.

08:46 Like, if you click on 403, for example, it talks about when it's used and when it's, like, how it compares to 404.

08:52 And then also how that reference code is set up as an enumeration in different languages.

08:58 So, for example, in Go, it's HTTP.status forbidden.

09:02 In Rails, it's colon forbidden.

09:04 And then Python 3, it's HTTP.htpstatuses.forbidden.

09:07 And things like that.

09:08 So, it tells you, like, the language version instead of typing it all in yourself and having magic strings and numbers everywhere.

09:15 Yeah.

09:16 So, it does look like, I don't know what Symfony is, but Symfony and Go do also support 418.

09:23 Status team.

09:25 Beautiful.

09:26 We're on par with Go in terms of the internet now.

09:29 Very good.

09:30 That was a good one.

09:30 It almost could have been the joke, but I like it.

09:33 All right.

09:33 Also, things that I like is that we get a chance to create amazing stuff to help people get better with Python out there.

09:40 So, over at Talk Python Training, we have three new courses coming very soon.

09:45 We have Getting Started with Data Science.

09:46 We have Moving from Excel to Python.

09:48 And we have Diving into Python Memory Management and Optimizing Your Programs around that.

09:54 All three of those are coming within probably a month or so.

09:57 So, if you want to get notified about that, just visit training.talkpython.fm.

10:02 And right at the front, there's a little email place you can enter your email address to get notified.

10:07 How about you?

10:08 I wanted to highlight Test and Code.

10:10 So, pytest 6 is out.

10:13 So, Test and Code is the other podcast I do at testandcode.com.

10:18 And I had talked last week that there was going to be a pytest 6 episode.

10:23 And instead of doing it just by myself, I contacted Anthony Sotilli and had him come on the show.

10:30 And so, we BS about stuff for about an hour.

10:33 Some of it's actual content.

10:35 So, I'm really trying to get – I'd really like to have the shorter episodes on there.

10:39 But, you know, I get to talking with somebody and it's just fun.

10:42 Just keeps going.

10:44 Awesome.

10:44 Yeah.

10:45 It's definitely a good one.

10:45 People should check it out.

10:46 What do we got next?

10:48 We talked about Pydantic.

10:49 Right?

10:50 So, Pydantic is a cool library.

10:52 You know, it's something I really just need to start using.

10:54 Like, I haven't created any new projects that probably really deserve this.

10:59 But I think it's definitely one of those things that I want to start using.

11:02 Because it's just so slick.

11:04 The ability to say, I have these models.

11:07 They have these types.

11:08 You can validate them.

11:09 You can auto-convert between them.

11:11 And so on.

11:11 Right?

11:12 So, we talked about that not too long ago, actually.

11:14 However, Andy Shapiro sent over a heads up about a new feature that's in the beta version of Pydantic that's coming.

11:22 That makes me pretty excited, actually.

11:25 I don't know.

11:26 We'll see how people feel about this.

11:27 But I'm excited about it.

11:28 I'm excited because I'm a fan of type hints.

11:31 Yeah, me too.

11:31 I love type hints.

11:32 Yeah.

11:33 I love them mostly because the way they make the editor help me write code.

11:39 Instead of going and saying, oh, how do I do this thing with this library?

11:44 Right?

11:45 If I just say the type of this thing is one of the objects out of that library and I hit dot, boom.

11:50 The editor shows me all the stuff I can do.

11:53 It just keeps me in flow and working on what I need to.

11:55 So, I love type hints mostly for that, but also for the validation.

11:59 One of the things they are like, but they do not do, is they are not like static typing in other programming languages where somehow that verifies what's being passed.

12:10 Right?

12:10 Yeah.

12:11 So, like, for example, if I have a function, it says it takes an integer, an age colon int, and that's what it says it takes.

12:19 But then I go and I write code and I give it quote seven instead of the number seven.

12:23 That's fine.

12:23 Python's like, yeah, that's cool.

12:24 You probably don't know what you're doing, but whatever.

12:27 We're just going to let it fly, right?

12:28 Well, with Pydantic, there's this new type annotation validator.

12:33 And so, what you can do is you can say, for this function, it has type annotations, and I want those to mean something.

12:39 Oh, okay.

12:40 Okay.

12:40 So, it gives it basically that it's not runtime behavior in the static languages.

12:44 It's a compile error, right?

12:45 Your argument string cannot be converted to enter whatever the compilation error happens to be.

12:51 But it would give you the runtime equivalent.

12:53 So, all you have to do is you have a regular Python function, and it has regular types, just like it would, you know, s colon str, count colon int, that type of thing.

13:04 And you can just work with it.

13:06 It's a little bit smarter than the compiler, maybe of, say, like C++ or C#, though, in that what you do is you just say, at validate arguments.

13:15 So, you give it a decorator that validates the arguments.

13:18 And not only does it validate, but it will convert if it can.

13:21 So, for example, I'll put an example in our show notes.

13:23 There's a function called repeat.

13:25 It takes a string and a number, and it's just going to echo out that string however many times that number exists, right?

13:32 So, you can say repeat, quote, hello, comma, three, and it'll print hello three times.

13:38 Super simple.

13:39 That passes the validation precisely because it takes a string and an integer.

13:43 But you can also say repeat, hello, comma, quote, four, and it'll still print out hello four times because it can take the string four and make it an integer four.

13:52 Oh, that's cool.

13:53 Right?

13:54 But if I say repeat, hello, comma, goodbye, boom, exception, validation error, you know, the value is not a valid integer, type equals type error integer or whatever, like some message there, right?

14:07 So, it says the count parameter is not and cannot be converted to the type specified by the type annotation.

14:14 Oh, yeah, this is cool.

14:15 How do you feel about this?

14:16 I obviously don't want runtime typing for all of Python code.

14:22 It's not, I don't want it.

14:23 But for special places, like maybe around your API.

14:28 Exactly.

14:29 The boundary of libraries, like stuff going in there.

14:32 Yeah.

14:32 You're like, please don't send me the wrong data.

14:34 Like, what is your other alternative, right?

14:37 If you really want to validate that, you have to do is instance of an int or try to cast it to an int.

14:44 And you've got to raise your own exception type and all that kind of stuff.

14:47 You're already going to have to do that work if you're writing a good API.

14:50 Or you're going to send out weird errors like int does not have, you know, type int does not have function split or some weird thing.

15:00 And people have to realize like, oh, what that means is you type some kind of, you send a number where a string value is accepted, right?

15:07 So this, if you really do depend on the types that you specify in your type annotation, this seems like a good idea to me.

15:13 The one drawback is this cannot make your code faster.

15:16 Right?

15:18 If it's doing both validation and type conversion in front of your code, it can't be faster than just not doing that.

15:25 But it might make your development faster.

15:27 It's not always about speed.

15:27 Exactly.

15:29 And it may make your sanity.

15:30 Yeah.

15:30 So one of the ways people have gotten around this in the past is instead of, is like for an example, doing, instead of trying to validate parameters to a function at an API level,

15:42 I have the parameters bundled into like an adders object and have adders validators written because adders does a...

15:52 Right, right.

15:53 Put it into a validated object type and then pass that thing over.

15:56 Yeah.

15:56 Yeah.

15:57 Gotcha.

15:57 Which is kind of what this is.

15:59 Yeah.

15:59 But without doing it.

16:00 This is cool.

16:01 Yeah.

16:01 I like it.

16:01 Yeah.

16:02 Very cool.

16:02 I like it too.

16:03 I was leaning Pydantically previously.

16:06 This makes me lean more that way.

16:07 I like it.

16:07 All right.

16:08 So one of the things that's really cool is, you know, we've had a lot of conversations about is Python fast.

16:12 Is it slow?

16:13 Is it fast for coming up with programs that work?

16:18 Does it execute fast?

16:19 Should you use libraries like Cython or rewrite some stuff in Rust?

16:23 But Anthony Shaw, he took it to another level, like further down than even something like rewrite bits in C or compile Python to C or something.

16:33 Yeah.

16:33 I think this is maybe an example of Anthony just having too much time in his hands.

16:38 I think maybe.

16:40 So Anthony wrote a, I just tried it this morning, a project called PyMult, P-Y-M-U-L-T.

16:48 You can pip install it.

16:49 And it just multiplies numbers.

16:52 And it only works for positive integers or negative integers.

16:56 It just doesn't do, it doesn't ever display negative numbers.

16:59 Anyway, regardless of that, it's an extension for Python written in assembly.

17:06 Because, why not?

17:08 And because of Anthony.

17:09 As in like MV and, you know, add like the assembly language.

17:14 That's the foundation of basically every other programming language?

17:18 Yeah.

17:19 In Anthony's Twitter announcement, he says, after a series of highly questionable life decisions, my Python extension written in pure assembly is now on PyPI.

17:28 It required an assembly extension for disk utils.

17:31 He also wrote a GitHub action support.

17:34 So it's running CICD and testing with pytest above and beyond, over the top.

17:39 But there is some coolness of it.

17:41 So it's a proof of concept to demonstrate how to create a Python extension in 100% assembly.

17:48 And then how to write a, you know, how to link those up.

17:52 How to call a C API and create a PyObject and parse PyTubule and stuff like that.

17:57 Basically all the stuff you have to do to get point parameters back and values out of an extension written in assembly.

18:06 And yeah, it's interesting.

18:08 So I like it actually.

18:10 Yeah.

18:10 And anyone wants to know what the code looks like.

18:13 It's like move racks comma X, imol, qword of Y, move result to racks, move eddy to result.

18:23 Like it's, yeah.

18:24 But then you get a call pylong from long, which is kind of a cool thing.

18:28 So there's this like interesting mix between the just pure assembly language and the C data types of CPython.

18:35 On a serious note though, anybody wanting to learn a little bit of assembly or something, it's not like often you need some sort of environment to try it out in.

18:44 Having some way to link Python and assembly is kind of neat actually.

18:48 So I'm grateful for that.

18:50 You're right.

18:51 Some of these commands.

18:52 I mean, I wrote some assembly in college.

18:55 That was a long time ago.

18:56 Yeah.

18:56 Well, you had to, Anthony, because that's some impressive stuff.

18:59 Yep.

18:59 Would you say it's easy though?

19:01 No.

19:02 No.

19:03 That's hard mode.

19:04 Yeah.

19:04 You know what actually is not as easy, I think, as it should be.

19:07 And it's honestly just bizarre.

19:08 Our properties in Python, as in at property, some function return some value converts what looks like a function call over to something that looks like field access.

19:19 Right.

19:19 Yeah.

19:19 But why does it have to have this bizarre sequence of conventions where I have an at property and then a function that has a name?

19:28 Okay.

19:29 That defines the name of the property.

19:31 That makes sense.

19:32 But then the subsequent thing, if I want to be able to set that, I have to say, you know, if the property was a, then I have to say decorator a dot setter and also have a function that is called a.

19:44 Like the a dot setter stuff is weird.

19:45 And then the order of the setter varies as well.

19:50 Right.

19:50 It has to be, or is constrained.

19:52 It has to be after the property a.

19:54 And what I've found also is that you can run into issues with inheritance.

19:59 I think if the property is defined in the base class, you try to create a setter on a drive class, things don't work right as well, which is all kinds of weird.

20:06 Right.

20:06 So there's just like, this is supposed to make life easy.

20:09 Why is this so complicated?

20:10 Like surely there was just another alternative that was like easier to do, to implement as well.

20:17 Anyway, it's always kind of been one of the bizarro things of Python.

20:21 That said, I love properties.

20:23 I love that you can do like lazy calculation with them.

20:27 You don't have to store them.

20:28 It's part of the memory.

20:29 If they can be recomputed from the values, there's a lot of good reasons to have them.

20:33 So Rude Vanderham sent over this cool project, which he created called easy property.

20:39 And do you know what?

20:41 It's easier than regular properties.

20:42 Okay.

20:43 I think that's why he named it that.

20:44 So the idea is, you know what you could do instead of have this property and then at a dot setter or a dot deleter is you could just say there's a at getter, at setter,

20:58 at deleter on the thing.

21:01 And it doesn't matter what order they are or if they're all defined.

21:04 Like for example, you can't have a setter without a getter.

21:06 Now conceptually, you should never really do that.

21:09 But just, you know, syntactically, like those things have to be defined there.

21:14 And like I said, the sort of base class, drive class type of thing, it gets wonky as well.

21:19 So with this, you just create a function.

21:21 The function name is the name of the property.

21:24 And then you put at getter on it.

21:26 And that makes it the kind that is, you know, the getter property.

21:30 If you want to have one, you can set, you say at setter.

21:33 The order doesn't matter or anything like that.

21:36 Nice.

21:36 Isn't that nice?

21:37 Yeah.

21:37 So there's a cool little example that he wrote.

21:41 You know, you can do them separately, which is probably what I would recommend and do.

21:45 But you can also have an at getter setter.

21:48 So you can have one function whose job is to both be the getter and the setter operation of the property.

21:55 And then the way it works is just the value is set to be like some default value, which is what happens in the, you know, in the get version.

22:03 You can have an at documenter.

22:06 You can have an at documenter.

22:07 So you can also have documentation for your property.

22:13 And there's a little function that will come up and do that.

22:16 So anyway, it's, it looks pretty nice to me.

22:19 It's not one of those things that's going to be, you're taking a heavy dependency upon to give it a try.

22:23 Right.

22:24 It's not like a, a runtime thing.

22:25 Like for example, that validate arguments, that's in between every time you call a function all the time, all the time.

22:31 Right.

22:31 So like this is the type of thing, like if it's going to work, it's going to work clearly, or it's just not going to work at all.

22:36 So give it a try and see if it makes you happy.

22:38 And if it sparks joy, you can have an at getter.

22:42 And I like the syntax better than properties.

22:44 Just more sense.

22:45 Yeah.

22:45 And you know, I think at property is just fine.

22:47 Like, okay, that totally.

22:49 But you know, at property name dot setter, like that just drives me.

22:54 It's just so bizarre and weird in the ordering and the dependencies of it existing in the same class hierarchy level.

23:00 Yeah.

23:01 Don't get me started.

23:02 So this is cool.

23:03 I like it.

23:03 Thanks for developing that and sending that over.

23:07 So we talked about assembly and properties and decorators and stuff.

23:11 So if anybody's left listening to this podcast, we do have more to talk about.

23:16 Let's talk about recursion.

23:17 No, just kidding.

23:18 I couldn't resist highlighting the last topic.

23:22 So there was an article by Ryan Howard that was on the test project.io blog called non-blocking assertion failures with pytestCheck.

23:31 And I had to highlight it because it's really the first time anybody's ever written an article about something I wrote.

23:36 So that's neat.

23:38 Yeah, very cool.

23:38 So pytestCheck is a library that allows you to do multiple failures within a test, mostly because assert normally uses asserts, but asserts stop after the first failure.

23:50 And sometimes you want more.

23:52 And I never really thought about the different use cases.

23:55 But so Ryan has the use case of using it with Selenium because sometimes when you're testing a page coming back, you might have to test multiple aspects of it.

24:04 His example was what's the content of some field and also if the URL is correct or something.

24:12 And I think with like a Selenium or some sort of web test, that totally makes sense because you'd have things like you want to check the error code and content and whether somebody's name shows up, then a whole bunch of stuff you could check and having multiple things.

24:28 Right.

24:28 You don't necessarily have just one assertion that captures all of the essence of what you're trying to determine, right?

24:35 Yeah.

24:35 And so I also linked to the pytestCheck library and then or plugin.

24:41 And then also an article that I wrote back and way back in 2015 where I started thinking about this and thinking how to solve it.

24:48 Okay.

24:49 Yeah.

24:49 Yeah.

24:50 Very cool.

24:51 Nice article, right?

24:51 And then I also wanted to do a public service announcement that because both Ryan and Anthony got this wrong, there are no capital letters in pytest.

25:01 So.

25:01 Hold on.

25:02 Hold on.

25:02 Hold on.

25:02 Let me try this.

25:03 Let me just.

25:03 Yeah.

25:04 Okay.

25:04 On this one.

25:05 But if I go over to Word and I type pytest is a testing framework, I'll bet you that Word makes it capital.

25:11 Yeah.

25:12 Doesn't that make it real?

25:14 No.

25:14 I know PowerPoint well.

25:17 So.

25:17 No.

25:18 Yeah.

25:18 That's always tricky, right?

25:19 When you have sort of a formal name, but the formal name doesn't have spaces or a slower case.

25:24 But I'm fine with it.

25:25 We got used to it with iPhone.

25:26 So there's no capital.

25:28 I mean, the P is capitalized in iPhone.

25:31 Yeah.

25:32 Will Word correct you, I wonder, if you try to type iPhone?

25:35 I would doubt it.

25:36 I mean, but that's Apple and they think differently.

25:38 That's what they tell me.

25:40 But there's others.

25:41 I mean, so there's some, I guess it's a weird thing with tools and stuff.

25:46 Some tools don't care whether you capitalize or not.

25:50 The people in pytest like it not capitalized.

25:53 So anyway.

25:54 Yeah.

25:54 Let's respect their lowercase p's.

25:58 Yeah.

25:59 All right.

26:00 So I have three really quick things.

26:02 They're all kind of fun.

26:03 So PyMC is a cool library for Bayesian analysis and probabilistic programming in Python.

26:11 So Alex, one of the core developers there, sent me a message, said, hey, we're planning the first ever PyMC on.

26:21 Okay.

26:21 I pronounce PyMC on because it's so on, they say.

26:25 And it's the first Bayesian community online conference around PyMC.

26:30 So if you are into probabilistic programming in Python, check it out.

26:34 There's a bunch of cool stuff they got going on.

26:36 They're also, I think, have a call for paper.

26:40 So if you want to do a presentation, shoot them a note.

26:44 And yeah, I'll link to that in the show notes.

26:47 Second, a while ago, we talked about Rumps quite a while ago.

26:51 Rumps is ridiculously uncomplicated Mac, I don't know, something, menu programs or something like that.

27:00 And so I ended up, I was sitting around, I have this little library that I run because so many things in my life require taking a title or taking some words and turning them into a file name or a URL name.

27:15 All right.

27:15 So suppose I've got the title of a video and I want to name an MP4 file that.

27:22 And it's got like a colon in it or like a forward slash or some random thing that shouldn't be in there.

27:27 I don't want spaces.

27:28 I want it.

27:28 So I wrote this little command line utility called URLify that I would just run.

27:32 It would take whatever's in the clipboard and replace it with that canonicalized version that would work well as a file name.

27:38 Oh, cool.

27:39 Yeah.

27:39 And that was cool.

27:40 But then I got tired of like always firing up the command line, the terminal.

27:44 Maybe it's busy doing something else.

27:46 So I got to do a new window, run that thing and then close it down.

27:48 And I was like, I just want to click something once.

27:52 All right.

27:52 I don't want to click the terminal and then start a new terminal.

27:55 And it was like not that much work, but you can imagine the way I'm going on and on about this.

28:01 I must be doing it more than is reasonable.

28:03 So what I did is in like 45 minutes, I converted that command line app to a Mac dot app file, like a full on dot app file that I could just ship to any Mac user.

28:13 They have no idea.

28:14 It uses multiple Python packages, a code that I wrote, and it runs as a GUI auto starting with my macOS when I log in in my menu bar.

28:24 Oh, wow.

28:24 Is that cool?

28:25 Yeah.

28:25 And so I linked to a little tweet that has a screenshot just says convert text.

28:30 But actually, I changed it to URL fi text, trim text.

28:33 So now I can just click up there and any text that I have, you know, if you copy something and you want to like paste an email or like, but maybe it's got a bunch of white space.

28:40 You're like, why is this all here?

28:41 Right.

28:41 Instead of putting in text editor and just getting the little bit out.

28:43 I just hit that and then I'll trim it or like a good lowercase text.

28:47 So I have these like little cool like clipboard text transforms and it all took like 45 minutes to turn it into a Mac app with pi to app and rumps.

28:56 Oh, wow.

28:56 So are you sharing this somewhere?

28:58 I will happily share it.

29:01 Okay.

29:02 But I don't have the GitHub repo public yet.

29:04 So let me just like put like a screenshot and what the heck it is and then I can make it public.

29:09 But yeah, so I'll put it in the show notes.

29:11 Neat.

29:11 Yeah.

29:11 Yeah.

29:12 It's super cool.

29:12 And finally, I just want to let people know who are taking courses over at Talk Python.

29:17 I was sitting around and I decided I want as you hover over the scrubber, I want it to show show a little thumbnail of where you would go in the video if you were to click the scrubber in that location.

29:30 What's the scrubber?

29:30 We're not like the little like time thing where you can click around and like zoom ahead and whatever.

29:35 And we're not like reusing Vimeo or YouTube or something that might potentially do that.

29:41 I think YouTube does it.

29:42 I don't know that Vimeo does.

29:43 Anyway, I recently figured out how to do that.

29:48 So if people are interested, they can check it out.

29:52 But there's a cool little trick where you can take a second hidden video player and point it at either the same video or a smaller version of the video.

30:01 And then as the mouse moves, just seek it to that time and show it above where the mouse is.

30:07 Isn't that crazy?

30:08 Yeah.

30:09 So if people need something like that, yeah, a little cool trick you might consider.

30:12 Not that often in Python, but if you happen to make your way to the JavaScript side of things, you'll do a lot of that, I guess.

30:18 So is that in your training site?

30:20 Has it been updated then?

30:21 Yeah, exactly.

30:21 It's not updated as we speak because I'm still busy transcoding 200 hours of video to 500 by 200 size.

30:31 So I'm about halfway through that.

30:34 I've had yesterday I uploaded 200 gigs of data.

30:38 I'm so going over my data limit this month.

30:42 That was 20% of my data limit in one day.

30:47 I got more to send.

30:48 So, but yeah, it'll be there as soon as the videos are done.

30:50 It'll turn it on.

30:51 I'm going to back up a little bit.

30:52 This IMC.

30:53 Yeah.

30:54 Probabilistic programming.

30:55 Can I use this to make a probability drive?

30:58 A probability drive?

30:59 Yeah.

30:59 Is this from science fiction?

31:01 I don't know it.

31:01 Like Hitchhiker's Guide to the Galaxy.

31:03 Oh, no.

31:03 Yeah, I would think so.

31:04 I would begin by feeding it the number 42 and see what comes out.

31:08 Maybe.

31:09 Yeah.

31:09 That's probably the natural first step.

31:11 Yeah.

31:11 All right.

31:12 Speaking of jokes, we've got two.

31:14 You want to go first?

31:15 Yes.

31:15 This was submitted by Reuven Lerner, inspired by Anthony Shaw.

31:20 I used to do low-level programming.

31:21 Then a product I bought told me no assembly required.

31:25 Since then, I've been coding in Python.

31:27 Nice.

31:28 Except for Anthony.

31:30 Anthony's coding in assembly.

31:31 He codes in both.

31:32 Yeah.

31:33 His Python is assembly code or something like that.

31:36 So this one, like last week, we talked about little Bobby Tables, who's beautiful, over at XKCD.

31:42 And I've got another XKCD for us.

31:44 But this one is not about databases.

31:46 No, it's about source control and Git.

31:48 Okay.

31:48 All right.

31:49 You want to be the woman developer that asks the question?

31:52 Yes.

31:52 I'll start us off.

31:54 All right.

31:54 So there's a couple developers speaking.

31:56 First one is talking about this new source control.

31:59 This is Git.

32:00 It tracks collaborative work on projects through a beautiful distributed graph theory tree model.

32:05 Cool.

32:05 How do we use it?

32:06 No idea.

32:07 Just memorize these shell commands and type them to sync up.

32:09 If you get errors, save your work elsewhere, delete the project, and download a fresh copy.

32:13 It's funny because it happens too often, I think.

32:21 Yeah.

32:22 I think most people have like the four or five or six Git commands that they use all the time and everything else they have to look up if they need it.

32:29 Yeah.

32:29 Merge conflict.

32:31 All right.

32:31 I'm deleting it.

32:32 I'm going to copy this back over.

32:33 Yep.

32:35 All right.

32:36 Funny indeed.

32:38 Well, thanks for being here as always.

32:39 And thanks to everyone for listening.

32:41 Thank you.

32:41 Bye.

32:41 Bye.

32:42 Bye.

32:42 Thank you for listening to Python Bytes.

32:44 Follow the show on Twitter via at Python Bytes.

32:46 That's Python Bytes as in B-Y-T-E-S.

32:49 And get the full show notes at pythonbytes.fm.

32:52 If you have a news item you want featured, just visit pythonbytes.fm and send it our way.

32:57 We're always on the lookout for sharing something cool.

32:59 On behalf of myself and Brian Okken, this is Michael Kennedy.

33:02 Thank you for listening and sharing this podcast with your friends and colleagues.

Back to show page