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


Transcript #86: Make your NoSQL async and await-able with uMongo

Return to episode page view on github
Recorded on Wednesday, Jul 11, 2018.

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

00:05 This is episode 86, recorded July 11th, 2018. I'm Michael Kennedy.

00:10 And I'm Brian Okken.

00:11 And Brian, we have a new special guest, Bob Belderbos.

00:14 Yeah, pretty excited.

00:15 Yeah, welcome, Bob.

00:16 Hey, nice to be here. Thanks.

00:18 Yeah, so people may recognize Bob from his PyBytes with an I, code challenge stuff that he does,

00:24 as well as the 100 Days of Code course that we did together, right?

00:27 Yeah, we did a major course on 100 Days of Code in Python. It was very fun.

00:34 Yeah, it was only, what, 20 hours of video and 80 hours of exercise?

00:38 Yeah.

00:38 That was awesome. So welcome to the show. We're super excited to have you here.

00:42 Before we get into everything, let's just say thank you to DigitalOcean, our customer,

00:47 and their sponsor of our show as well. Let's check them out at pythonbytes.fm/DigitalOcean.

00:52 You'll get $100 free credit if you're a new user. That's pretty awesome.

00:56 Brian, if I had to guess, if I were just going to pick out of thin air what kind of topic you might have,

01:00 it might have something to do with testing.

01:02 Yeah, I do like testing quite a bit.

01:05 So what do you got for us?

01:06 Well, we have responses, and there's a funny story around it, but this is,

01:11 responses is a utility library for mocking out requests, Python library.

01:17 And it's from GitSentry.

01:20 So it's under the GitSentry GitHub account.

01:23 And it looks pretty cool.

01:25 It's actually, we'll include a small snippet in our show notes, but there's,

01:30 it's a decorator approach to, and other ways to be able to test your, pretty much mock out.

01:37 If you have an application that uses requests to access any, really any end point on the internet,

01:43 you can use this to mock out those.

01:45 And it has some dynamic, dynamicness to it.

01:49 And it's, it looks actually pretty complete and pretty nice.

01:52 That's pretty sweet.

01:53 Yeah.

01:53 You put an attribute on your, your function.

01:56 And during that one, any call to requests like request.get or the response.json or whatever

02:02 is going to be basically overridden as if you had done that with patch, but it's, it's a little more focused on just requests, right?

02:09 Yeah.

02:09 It's, it's focused on requests and it's also, it's designed to go well with the pytest framework.

02:16 So it cleans up after itself really easily.

02:19 Nice.

02:19 How does it integrate with pytest?

02:21 Like what, what's special about that?

02:22 You have the decorator you put on your test is like responses activate.

02:26 So it'll, and then you're, you're going to have it, you register a, an end point to

02:32 mark and what response it should give.

02:34 And then all of that gets cleaned up at the end of the test.

02:38 You don't have to undo it or anything like that.

02:40 Yeah.

02:40 That's awesome.

02:41 Very nice.

02:42 The funny thing is, is this is exactly what Anthony and I were, we're considering building

02:48 together.

02:48 We just didn't know this one existed.

02:50 And so I wanted to point it out because, your responses is a clever name to go with

02:56 requests.

02:56 I get it.

02:57 But if I'm trying to find something that goes well with pytest and goes well with, requests,

03:03 I might not know to search for responses.

03:06 So that's why I'm violating it out.

03:08 So I think anybody that's trying to do something like this should check it out.

03:12 It looks pretty complete.

03:13 Cool.

03:13 It looks very useful.

03:14 So it's not making any call over the network, right?

03:17 That's the whole point of, of patching it out.

03:20 Right?

03:20 Well, it's patching.

03:22 It'll patch out the, the end points that you specify.

03:25 I don't really know.

03:26 I don't know what the behavior is for other ones.

03:28 It probably lets everything else go through.

03:30 Yeah.

03:30 That's the one of the things I like about it.

03:32 As you say this, if I request this HTTP endpoint, then do this rather than I'm just replacing,

03:39 you know, request.get with everything.

03:41 Right.

03:42 Right.

03:42 Oh yeah.

03:43 Pretty nice.

03:43 I do see your point about SEO and discoverability for responses because like response and responses

03:50 is just such a common word on so many API libraries.

03:53 Like how are you going to find the testing one?

03:55 Yeah, I was looking.

03:58 It says in its description that it's Python 2.7 or newer.

04:01 Well, so it isn't obvious to me that it was 3x, but it looks like in the investigate the

04:08 toxiny file, it's tested up through 3.5 at least.

04:12 So it's probably 3.6 and 3.7 compatible too.

04:15 Haven't tried it, but.

04:16 Yeah, probably.

04:17 Probably.

04:17 Yeah.

04:18 Very nice.

04:18 All right, Bob, it's time for yours.

04:20 What one did you bring?

04:21 Yeah, I found an interesting decision tree article, 29 common beginner Python errors on one page.

04:29 And I found this interesting because these kind of decision trees or graphics, you get

04:35 a lot of information in one page you can print out or study in detail.

04:40 And it's like a flow.

04:42 It's like an infographic for like getting my code to work.

04:45 It's amazing.

04:46 Yeah, it didn't land directly on the article.

04:48 Actually, it was Microsoft devs that tweeted it out.

04:51 And that's how I found it on the Python hashtag.

04:54 And it goes through this whole flow, like what kind of error do you get?

04:59 And then it goes into these common Python errors, like attribute errors, syntax error, type errors.

05:05 And for each one, it gives like a couple of plausible causes.

05:08 And yeah, I mean, if you have more experience, some might be obvious, but some are very subtle.

05:14 And that was like a great reminder, like comparing three int with a three string that returns false.

05:21 And for example, when I was going through it, I also could relate to similar things.

05:28 Like when you exhaust the file, that also can happen with a generator, right?

05:32 In generator, you go through it once.

05:34 And then if you do that again, then, hey, it's like an empty list because a generator only, you can only consume once, right?

05:42 So it was a nice, I guess, exercise to take a lot of info in from one infograph.

05:48 And I think for a beginner, it's very useful to go through that at least once.

05:52 And I think the author made the infograph after having teached Python for a long time in the biology space.

06:01 So that was also inspiring for me, like teaching others.

06:05 Yeah.

06:06 How you could summarize that information at some point.

06:09 I'll try to give a quick summary just so people can get the visual.

06:12 So it's like an infographing.

06:13 It says, start here.

06:14 Do you get an error when you run your code?

06:16 Yes.

06:16 What type of error do you get?

06:17 And there's a whole branch there.

06:18 No.

06:19 Does your code use loops or if statements?

06:21 If you use an if statement, go check this.

06:23 It just like walks you through some pretty basic stuff.

06:25 I kind of like it.

06:26 Brian, what do you think?

06:27 Yeah.

06:27 Actually, this is one that I've run across before.

06:31 And when our team started using Python for testing, one of the other engineers actually printed this out and posted it on the side of their cubicle so that people that came and asked them questions could just look at that first.

06:43 Oh, that's pretty cool.

06:44 And I like the fact that it's coming from Python for biologists.com.

06:49 So we're seeing Python being taught in a lot of different domains, which is awesome.

06:53 Yeah.

06:54 That actually is a really good point.

06:55 It's pretty awesome.

06:56 So the next one that I have for us is this thing called MicroMongo, which somehow I have not heard of previously.

07:03 But Brian, you know, I go on and on about async programming stuff and parallelism and things like that, right?

07:10 Yes.

07:10 And I do know that you kind of like MongoDB also.

07:13 This is like I get to put them together.

07:17 It's amazing.

07:18 So one of the main challenges of this async await stuff actually becoming super useful and super powerful.

07:26 There's two fundamental problems limiting like the serious adoption of this.

07:30 One is the web frameworks don't deeply support it.

07:33 So I can't have an async flask method.

07:36 I have to switch to something like Cort or use Fibora or some other framework like the main Django pyramid flask.

07:41 They don't support the async.

07:43 So that's thing one.

07:44 But we're seeing some movement there.

07:46 But even if you could get that to work, a lot of the ORMs don't have great ways to take advantage of their parallelism or their potential parallelism while they're waiting on the network talking to the database.

07:58 Because usually that's what your web app is doing and that's why you would want the async.

08:01 It's like you could let it do other processing while you're waiting on a database response.

08:04 So what you need is if you're using an ORM or something equivalent called an ODM for Mongo is the ability to interact with those databases in an async way.

08:15 And so this micro Mongo is a very lightweight and small, as you might imagine from the name, object document mapper for MongoDB and Python that brings this ability to do async operations against MongoDB.

08:27 Right.

08:28 So if you wanted to use, say, Cort or Fibora and you want to use MongoDB as the backend, you have to use something like this, which is pretty sweet.

08:35 Very interesting.

08:36 I need to go on the async train myself yet.

08:39 So maybe this is a good opportunity to learn it.

08:41 So definitely.

08:42 It's not that, I mean, Mongo itself, you don't need, that's on the other end.

08:47 So that's already asynchronous or can be, but it's the document mapper that's the problem.

08:53 You need the library that talks to the database to have an async option.

08:57 So for example, like let's take something that people know more about probably SQLAlchemy, right?

09:02 If you go to SQLAlchemy and you create a query and you say like order by or filter or whatever, you need SQLAlchemy to be rewritten.

09:10 So that filter is async is an async method or it's useless in terms that, right?

09:16 So it's like the place you want to get to is interacting with your database asynchronously.

09:20 But if the ODM or ORM that you're working with in the middle doesn't support that, there's zero workaround.

09:26 You're done, right?

09:27 You cannot do that.

09:28 Whereas most of the things you're doing is waiting on either web services or you're waiting on databases.

09:33 And that's where you would get like the massive scalability.

09:36 Another thing that's really cool here is there is something called motor, which is another async way of programming against MongoDB.

09:42 But this one will let you choose the foundational bits.

09:46 So you can say, I don't care about asynchronous programming.

09:49 So I just want the base to be PyMongo.

09:51 Or you can have TXMongo.

09:53 Or you can use motor async, which is something from Jesse Davis, I believe, at MongoDB.

10:00 Or you'll like this, Brian.

10:01 You can even say the driver, the foundational database access part of this ODM is MongoMock.

10:08 So if you want, you can just say actually replace the internals with this mocking database layer for testing.

10:13 Oh, actually, that's pretty cool.

10:15 Okay.

10:16 Neat.

10:17 Yeah, so this is, I haven't used it.

10:19 I generally use Mongo Engine.

10:21 But this is really looking quite promising and powerful.

10:24 So definitely want to give a shout out to that.

10:27 Speaking of shout outs, let's give a shout out to DigitalOcean.

10:29 So DigitalOcean is sponsoring this episode.

10:32 And they have a great, very affordable, very reliable service over at DigitalOcean.com.

10:39 So our stuff runs on DigitalOcean.

10:41 We have a couple of servers doing all the magic, including one doing MongoDB.

10:45 Back there somewhere in New York City, I think it is.

10:48 Anyway, they're super great.

10:50 Go over there and create a virtual machine.

10:51 Get an awesome machine for five bucks a month.

10:54 Get it set up in 30 seconds.

10:56 And off you go.

10:57 If you want to create something more pre-configured, they have a bunch of one-click apps to say set

11:02 up like a ghost blog for five bucks a month for one click.

11:05 Things like that.

11:06 So check them out at pythonbytes.fm/DigitalOcean.

11:08 You get $100 credit for a new user.

11:10 And you'll let them know you're supporting the show.

11:13 And thanks for them supporting the show.

11:15 So, Brian, what's up with this statistics stuff here?

11:19 I've not done statistics for a really long time.

11:21 I'm one for saying that I've been known to oppose the stringent math requirements in a lot of computer science and other science fields.

11:30 But regardless, there's a lot of people that end up having to learn statistics.

11:36 And I think statistics is a good thing to learn, at least some of the basics.

11:40 Yeah, I think we often come into, like, even going through computer science, you might not even take statistics.

11:45 You probably took calculus and differential equations and other stuff that you never, ever use again.

11:49 Right.

11:49 Yeah, exactly.

11:50 Yeah.

11:51 Let's not go down that tangent too far.

11:54 We'll all start crying about remembering how much work we put into learning that stuff, right?

11:58 I know I have never needed to factor a polynomial, ever, in real life.

12:03 How about computing the inverse of a non-segular matrix?

12:06 Sorry, let's keep going on statistics.

12:07 However, sometimes I do need to know, like, averages and medians and stuff like that.

12:14 So this is, I'm highlighting this article.

12:16 It's called the Basic Statistics in Python, Descriptive Statistics.

12:21 And I like it, it's just using some simple Python to demonstrate how you would build up some simple statistics concepts to teach the concepts, like min, max, mean, median, mode.

12:35 Some of the others, like standard deviation.

12:37 It's just a handful of them.

12:39 But those are kind of hard things to get your head around when you're first looking at it.

12:43 So being able to play with the numbers with a computer, I think, is a neat thing.

12:46 So they give some descriptions on how you would implement some of these functions in Python.

12:52 Oh, yeah, that's nice.

12:53 It does a nice job.

12:55 Like, for instance, if you wanted to figure out the average, you can take the sum of all of everything and then take how many scores.

13:04 And it's like the sum divided by how many.

13:06 Those are good concepts to look at.

13:08 The thing that was missing out of it is the little bit at the end, I think, that says, by the way, don't ever do this if you're actually writing Python code.

13:17 Because all this stuff's already built in.

13:19 Exactly.

13:20 Yeah, you've got the various numerical libraries for it, right?

13:23 Isn't this stuff already built in?

13:25 And, you know, min and max are already built in just normally and some of the others.

13:29 But I actually went and looked and found I didn't know about the statistics module.

13:35 That's built in, I think, as of 3.4 or 3.5 or something.

13:40 That's right.

13:40 It's quite new.

13:41 I don't remember.

13:42 But I think it might be 3.5, actually.

13:44 3.4.

13:44 3.4.

13:45 Thank you, Bob.

13:46 Okay.

13:47 But I threw in some examples.

13:49 But, you know, if you're just using that, it doesn't help you learn what statistics is.

13:55 So that's why I think the article is still good.

13:57 I stuck up.

13:58 I was looking at the statistics and I noticed that it had both standard deviation variance and population standard deviation and population variance.

14:07 I don't know enough statistics to know what the difference in those are.

14:12 I know P goes in front of the method name.

14:14 No, that's awesome that that's in there.

14:17 Yeah, I think it's interesting that they added that module into the standard library in 3.4.

14:22 It's cool.

14:22 Yeah, I guess maybe they were tired of people implementing them in crazy hand-coded ways.

14:28 Yeah, and it's got to be faster written in C to compute the variance than to do it in, like, a Python loop.

14:33 That's a good point.

14:34 Yeah, and it, again, goes to say that always keep up with the standard library because there's so much stuff in there that can save you writing code and probably gain you performance.

14:44 Yeah, absolutely.

14:45 All right, Bob, what's the next one you got here?

14:47 This is a big piece.

14:48 Strings and character data in Python by real Python.

14:52 I don't have the...

14:54 If it's by real Python, you know it's kind of like a novella.

14:57 Yeah, exactly, because I want to compare it with that AetherTools article you featured a couple of weeks ago.

15:05 That was, like, this long piece that took me an hour to read, but I got so much out of that.

15:09 And it's the same here.

15:10 I didn't even, honestly, didn't finish it yet.

15:13 I'm, like, 60, 70% in and already got so much value out of it.

15:17 Because it does this great detailed tour of strings in Python and all the methods you can use on them.

15:25 And I put some snippets in the show notes because, I mean, for a beginner, it's great because you will be working with strings from day one.

15:33 But even if you know Python, there's, like, a lot of these little tricks in there that can just make you faster in Python.

15:41 If a string is a digit, you can use isDigit on the string, which returns a Boolean.

15:47 Rather than try to convert it to an int, catch the exception, go, looks like it's not an int.

15:52 Yeah, that kind of thing.

15:53 Yeah, so you can use that method.

15:55 Or, for example, you want to look for white space, a space, a tab, or a new line.

15:59 There's also an isSpace method that does that for you.

16:03 Or commonly with terminal apps, you want to have this banner with the text centered.

16:09 Well, you can just do string, center, and then the width, and then the filler character.

16:16 So there's a lot of good stuff in there.

16:19 And I think it's worth to go and spend that hour and learn all that stuff.

16:23 Because, I mean, it will just shave off time when you're actually coding.

16:27 And another thing I also found, kind of the linking you do when you read such an article.

16:32 For example, polymorphism, right?

16:35 If you have the index method, it works on a string, but it also works on a list.

16:39 So I put a snippet in the show notes that I use index on a string and a list.

16:45 And you see that they behave in a similar way.

16:48 Same goes for count.

16:49 So I found that also an interesting point to highlight.

16:53 Oh, yeah.

16:53 This is really interesting.

16:54 I just threw that into Instapaper.

16:56 And Instapaper says, reading time, 33 minutes.

16:58 Yeah, this is definitely, it's like a book chapter almost on strings, character strings.

17:03 I might be a slow reader, but.

17:04 Well, I'm sure I would, too.

17:06 And, you know, that's not counting that it's like code.

17:08 Code you got to analyze, right?

17:11 Yeah.

17:11 You want to try some things.

17:13 Yeah.

17:13 Yeah.

17:13 Yeah.

17:13 Exactly.

17:14 Exactly.

17:15 That's a great find.

17:16 It's really cool.

17:17 And people coming from different languages, I know I was like this.

17:20 I did not expect that strings would have all of these cool operations on them.

17:26 Yeah.

17:26 Especially if you're coming from C, where strings are really just, you know, character, pointer, array type things.

17:32 They're basically just memory, right?

17:34 Yeah.

17:34 Yeah.

17:35 But imagine you have to do that, leaving zeros to a number.

17:39 And that would take you a couple lines of code.

17:41 In Python, it's just a number, string, z fill, and then the number of zeros you want to be penned.

17:47 And it just makes for shorter code.

17:48 Yeah, it's wonderful.

17:49 Z fill?

17:50 Who's he?

17:51 Yeah.

17:51 Yeah.

17:51 Yeah.

17:51 So, you want to have 42, for example, and it needs to be a width of five.

17:57 You can just set fill it with zeros.

18:00 Nice.

18:02 So, I'm going to close this out with something that is surprisingly controversial.

18:07 I don't know.

18:09 I'll see what your thoughts are.

18:11 So, there's a new PEP in town.

18:13 So, there's a new PEP in town.

18:13 Pep 572.

18:15 And whenever I think about these things, I always think, you know, someone has worked super hard on this.

18:20 And that's really awesome.

18:21 They're making a contribution.

18:23 And then, you know, to see people react in, I don't know, not totally excited ways.

18:28 I don't know.

18:29 It's sort of, I don't know how to feel about it, let's say.

18:31 But there's a new piece of Python syntax that will allow you to, in a single expression, create a variable and assign to it.

18:41 So, so often you have to do something to the effect of, like, I'm going to create a variable, set a value, and then if that value is something, I'm going to do a thing or I'm going to do something else.

18:55 So, the new syntax lets you put a colon equals to define that and put it all in one line.

19:01 So, instead of saying, like, match equals pattern.search, if match is not none, et cetera, et cetera, you can just say, if match colon equals pattern.search is not none, all in one.

19:12 I'm actually even confused about your example.

19:14 I thought that the point to that new operator was explicitly to check against none.

19:20 The point of the thing is to allow you to both create the variable and set its initial value and check it at the same time, whatever you're checking it for.

19:29 Oh, right.

19:31 And you don't even have to check it.

19:32 For example, one of the really common use cases for this is to use it in, like, a list comprehension or something like that.

19:39 And any time you need more than one line, that cannot be put into a list comprehension or a lambda expression, right?

19:47 But now, with this one line variable creation assignment potential test behavior, you can put these into inline expressions like list comprehensions or lambdas or so on.

19:59 I think that's it.

20:00 I've only skimmed some of the use cases, but that's what I'm thinking it's for.

20:03 Okay.

20:03 So this is interesting.

20:05 So this is interesting.

20:05 The syntax is not terrible.

20:06 I think it's kind of okay.

20:08 There's already lots of examples of this in other languages that had pretty decent implementations.

20:13 So one of my thoughts was like, well, why didn't that get adopted?

20:17 So both in like C# link expressions as well as JavaScript expressions, you have let something equals a value in terms of like a, like in terms of a for loop or something like that.

20:27 That could have also been an interesting option.

20:29 But I don't know.

20:31 To me, it's okay.

20:32 But I don't know that it's necessarily needed for the language.

20:35 Like it's not solving a problem that I think very many people actually have.

20:39 I wouldn't have to use it and see it in other people's code because from the snippet you posted, I find it a bit confusing.

20:46 It might just be me having to get used to it.

20:50 But it's not the kind of syntax I'm used to from Python, right?

20:54 Which is more readable.

20:55 What do you guys think?

20:57 To me, it's like one step down the slippery slope.

21:01 You know, I mentioned C# and the let stuff.

21:03 And I feel like the C# language used to be pretty nice.

21:06 And it's just a complete train wreck.

21:07 Now, there's all these little three or four ways of doing the same thing.

21:11 And they're all getting like a few characters shorter.

21:13 But now there's five of them.

21:14 They're all more or less equivalent.

21:16 It's just like, whoa, why do we keep adding these things to this language?

21:20 They're just unneeded.

21:21 And I feel like this is sort of in that category of stuff.

21:24 Python's nowhere near as bad.

21:25 Might not be the PEP itself, but more what we start to introduce and what other syntax changes

21:33 that might be coming, right?

21:34 If you open it up for this.

21:36 Yeah.

21:37 So what I want to sort of point people at for this is the tweet by Raymond Hedner.

21:42 And then the Twitter conversation that follows it, which is really interesting.

21:47 Like you don't normally see this much conversation in a thread on Twitter.

21:51 But there's like 44 messages onto this one tweet.

21:55 And it's pretty interesting.

21:57 Some people like it.

21:59 Some people don't.

22:00 But I think it's worth reading through.

22:01 Nice.

22:02 So anyway, new PEP in town.

22:03 And it's approved, right?

22:04 This is not proposed.

22:05 My understanding is this is approved.

22:07 So it's now going to be part of maybe Python 3.8.

22:10 Wow.

22:10 Yeah.

22:11 I'll totally use it.

22:12 But it is.

22:13 It did one.

22:13 It was one thing that made Python different was that like in C, if you tried to do check

22:20 for equality and you accidentally did assignment instead, it's wrong.

22:25 And I guess that's why the syntax is specific so that it's not going to be an accident.

22:30 Right.

22:31 In C, you could say something to the effect of like, if match equals pattern.match, do

22:37 such and such.

22:38 Right.

22:38 And you actually meant if those two things were actually equal.

22:41 Right.

22:41 So there's a way.

22:42 The syntax does make that sort of fall through error case not actually.

22:47 You can't omit an equal on accident and actually get assignment.

22:50 Yeah.

22:50 You have to explicitly put the colon in there.

22:53 It's colon equal.

22:53 So.

22:54 Mm-hmm.

22:54 Okay.

22:55 All right.

22:56 Well, there it is.

22:57 PEP 572, the most controversial accepted PEP I've heard in a little while.

23:01 Pretty interesting.

23:01 All right.

23:02 Well, that's it for our news items, you guys.

23:04 Brian, you got anything else you want to share with the world these days?

23:07 What's going on?

23:08 Well, actually, kind of nice tie-in with this PEP 572.

23:12 I wanted to just talk with somebody about all the new stuff that's actually proposed and

23:19 accepted already for 3.8.

23:21 And I think just for 3.8.

23:24 So that's kind of what Anthony Shaw has been up to lately.

23:27 So I'm going to have Anthony on testing code and we're going to talk about that.

23:31 Oh, that's going to be great.

23:32 I've also realized that doing this podcast with you, it's pretty easy to do it once a

23:38 week because you're waiting for me to do it once a week.

23:40 So I'm trying to rope in some other people and Anthony's one of the people that agreed

23:46 to try to do testing code more regularly to try to get more of those out.

23:50 Oh, yeah.

23:51 That's awesome.

23:51 Yeah.

23:52 That's really cool.

23:52 How about you, Michael?

23:54 Well, if we want to give shout outs to Anthony, I just had him on Talk Python as well about

23:58 his security article.

23:58 He's really killing it.

23:59 Him and the real Python guys, they're definitely cranking out the content.

24:03 That's great.

24:03 But the thing that I guess is the biggest news that has me getting up early and staying up

24:08 late is I'm working on a new about 10 hour course for data driven apps in Pyramid with

24:14 SQLAlchemy and like production migrations and all sorts of stuff.

24:17 That is almost done.

24:19 It should be out next week.

24:21 I'm just finalizing a lot of the videos this week and it'll be rolling.

24:25 Nice.

24:25 Nice.

24:26 Nice.

24:26 So that's what I'm up to.

24:27 Bob, how about you?

24:29 You want to tell people about PyBytes real quick?

24:31 What that is?

24:31 Yeah.

24:32 PyBytes started out as a blog with articles and we quickly moved into a blog code challenges

24:38 and that got quite some traction.

24:40 So we built out a code challenge platform, which you can find on codechallenge.es.

24:46 So code challenges spelled as code challenges together, but ending in .es.

24:51 And you can log in with GitHub and code in the browser.

24:56 So there are exercises with automated pytest.

24:59 Yes, Brian, it's pytest.

25:02 And you can code in the browser and verify those exercises.

25:05 And yeah, the feedback is great.

25:07 I mean, people are learning a lot of Python that way because it's so practical.

25:12 And so we're growing that.

25:14 We have a Slack community behind it where a lot of people are joining and we have very good

25:19 Pythonic discussions.

25:21 So let's, yeah, I'm excited about this.

25:24 Yeah, right on.

25:25 Yeah, it's a cool platform.

25:26 Happy to see you doing it.

25:27 And thank you for being on the show, Brian.

25:30 Thank you.

25:31 Thank you, as always.

25:32 I'm happy to hear you're doing more testing code as well.

25:34 That's pretty sweet.

25:35 Yeah.

25:35 All right, catch you later.

25:36 Bye.

25:36 Thank you for listening to Python Bytes.

25:40 Follow the show on Twitter via at Python Bytes.

25:43 That's Python Bytes as in B-Y-T-E-S.

25:46 And get the full show notes at pythonbytes.fm.

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

25:53 We're always on the lookout for sharing something cool.

25:57 On behalf of myself and Brian Okken, this is Michael Kennedy.

26:00 Thank you for listening and sharing this podcast with your friends and colleagues.

Back to show page