Mastering API Testing with FastAPI: Databases, Dependencies, and More!

  Рет қаралды 78,996

ArjanCodes

ArjanCodes

Күн бұрын

Пікірлер: 108
@ArjanCodes
@ArjanCodes Жыл бұрын
👷 Join the FREE Code Diagnosis Workshop to help you review code more effectively using my 3-Factor Diagnosis Framework: www.arjancodes.com/diagnosis
@nathanbrown2640
@nathanbrown2640 Жыл бұрын
In terms of other ways to handle database testing: Rather setup and teardown of the entire database contents each time you run the tests, we have a Docker Image for our test data. That means we can test against a consistent dataset each time, and easily reset it (i.e. destroy the container) ready for next time (spin up a new container from the image). This works well for the things like: - We have a very large schema and set of testing data - Handles the fact we are still more monolithic than microservices (when/if we can fully split out, we would need separate images for the different services) - We want to test against specific MySQL versions (i.e. testing against 5.7 with one image and 8.0 with another image) - Standard Docker benefits of being platform independent, and indeed installation independent (i.e. our server can run the tests on Linux whilst devs can run happily on Windows) We store the Dockerfile with the schema and initial data in a repository that therefore allows versioning, plus new data can be added as a new file at the end of a list of scripts to be run to setup the data. Thoughts? Are there any particular weaknesses of this approach? Or better benefits of other methods?
@erikvanraalte4557
@erikvanraalte4557 Жыл бұрын
To me it seems that destroying your containers and subsequently restarting it for each test, doesn't scale so well. The first question that pops up is: Do you really need all that data in your db for each of your tests? Far from an expert here, but: - You could have a single session for read only tests, as they are not making mutations to the database (and hence no need to reload or reprovision). - Like I said earlier, ask yourself if you need all the data in the first place - as part of a pytest fixture, you could copy the OG database to a unique file name, and passing that to the testclient/test, hence yielding a fresh playground db for each test, all without needing to restart your container. Of course it is different when you are testing on API level, in that case your approach makes a lot more sense already. Some (dirty) tricks that I built into my application (when in test environment) is a seperate API, to load or alter databases in my application, hence giving me more control when I run e2e tests.
@djpaulytee
@djpaulytee Жыл бұрын
@@erikvanraalte4557 Why not initialize all the data the beginning of the test suite and run each test in a transaction that rolls back after each test? If your tests require transactions you can use nested transactions.
@TheRich107
@TheRich107 Жыл бұрын
Do you mean docker image or docker volume with the db data on ? For those with smaller database fixture needs... Factory boy library is your friend
@nathanbrown2640
@nathanbrown2640 Жыл бұрын
​@@erikvanraalte4557 Good point - that is the question I need to ask but have probably been ignoring as it is a bit of hard one! I should clarify that we spin up the container only once for each test run (i.e. once for 2500+ tests, and growing as our coverage increases). However, I'm starting to run into problems where the data change from previous tests can affect later tests, so your scaling point does hold a lot of weight too. It's leading me to think we'd need to rewrite the tests (e.g. why do we care about the next id in the table being a certain value?) Or to split the suite up and spin down and up between these sections. Still feels like our method isn't great, but is a starting point... I like the idea of refining/splitting up the data better, plus it makes sense to have a cleaner dataset when testing anyway to be very clear that we are testing what we think we are testing.
@nathanbrown2640
@nathanbrown2640 Жыл бұрын
​​@@TheRich107Thanks, that looks excellent!
@gweb214
@gweb214 Жыл бұрын
I don't know if it is on purpose or not, but you have been timing your video releases with my Friday lunch breaks and I love it.
@sepehrghafari428
@sepehrghafari428 Жыл бұрын
Thank you for the practical videos that can be used in real world work projects. The writing test could not be explained in detail more easily than this.
@ArjanCodes
@ArjanCodes Жыл бұрын
Thank you! I'm glad it was helpful :)
@andreramalho9908
@andreramalho9908 Жыл бұрын
I’m a simple man, Arjan posts a new video, I watch it! Congrats for the good work, Arjan!
@ArjanCodes
@ArjanCodes Жыл бұрын
Haha, thank you!😊
@mikhail349u
@mikhail349u Жыл бұрын
It's better not to create items needed just in some tests in the common setup. Every test should be atomic and work with an empty database. So if you need an item for an update test, you should create it within that test
@estevaofay
@estevaofay Жыл бұрын
Agreed
@kt4760
@kt4760 Жыл бұрын
Pytest-sqlalchemy-mock creates a "mocked_session" fixture which can be used in tests. Also useful is that mocked tables/records can defined as lists of dicts.
@Bubblemaker1987
@Bubblemaker1987 Жыл бұрын
A very good practical guide to testing in python in general, including some comments on integration tests. Thanks!!
@ArjanCodes
@ArjanCodes Жыл бұрын
I'm glad you enjoyed the content!
@FransjeFranchise
@FransjeFranchise Жыл бұрын
Love how you introduced the concepts incrementally. Any plans to cover SQLModel? It plays very nicely with fastapi
@DeKeL_BaYaZi
@DeKeL_BaYaZi 7 ай бұрын
Dude the way you explain stuff is out of this world
@ArjanCodes
@ArjanCodes 7 ай бұрын
Thank you so much! Glad you enjoyed the video.
@elliotcossins8417
@elliotcossins8417 Жыл бұрын
Your videos are amazing, you are an amazing teacher, please never stop making them!
@ArjanCodes
@ArjanCodes Жыл бұрын
Thank you for the kind words, Elliot! Will do!
@timbrap4693
@timbrap4693 Жыл бұрын
Would love to see async tests with sqlalchemy
@po6577
@po6577 Жыл бұрын
Since it’s testing. I suggest you simply use not async version to test. Same as alembic. So you will have async db and stuff in your prod environment and not async while doing migration and testing. Wonder why you want async with testing?
@TheRich107
@TheRich107 Жыл бұрын
Speed? For me I used xdist. Database stuff wise you get issues so you need separate databases (or schemas) for each pytest session. This allows you to run close to X time the number of CPUs you have. Downside is the set up time so I also have a single pytest runner when running single tests. Config for creating a database client that uses separate sessions per thread goes in the conftest file.
@timbrap4693
@timbrap4693 Жыл бұрын
@@po6577 I get what you are saying. The problem I encounter is that when I want to test my endpoints, they are all written with async functions and methods. Meaning that the way I handle my CRUD operations is async. I have never tried this up to this point of writing, and i'm not sure why now that I think about it. But since all of it is using async sqlalchemy, I can't use a dependency override that inserts a sync_engine. (Will test this later, but I assume it's incompatible). So to actually test my endpoint, I need to setup everything async with pytest. Now I have been able to do this, but how I'm not ENTIRELY sure how I did it, so I consider it a hack.
@TheRich107
@TheRich107 Жыл бұрын
Not sure I follow, The controller can just be hit with a test client. That will not care if your function is sync or async. Not tested async service layer before though is that what you mean ?
@ripichipina
@ripichipina Жыл бұрын
Nicely done! Short but very solid and specific one. Thank you very much!
@ArjanCodes
@ArjanCodes Жыл бұрын
Thank you for the kind words! I'm glad it was helpful.
@unperrier5998
@unperrier5998 Жыл бұрын
Another issue with this approach of using sqlite is database compatibility: I had a similar setup and sqlite did not support ARRAY, which is very common in postgresql. The solution I came up with is to write a dialect adapter for sqlalchemy that emulates the ARRAY command. But that custom code may have bugs that hide real bugs (false positives) and you don't want to start testing the test framework :)
@TheRich107
@TheRich107 Жыл бұрын
Docker makes it so easy to spin up a real database that for advanced features it makes sense to us the real thing. Great for when You need to test your app on the next database version or library. Bumping from sqlalchemy v1 to V2 for example.
@robsnook4512
@robsnook4512 Жыл бұрын
@@TheRich107 Yep, I just have it on create, it will spin up a mariadb instance, create all the tables, Once it's done, i can just clone the production database to it then i can be as destructive as i want.
@pramodjingade6581
@pramodjingade6581 Жыл бұрын
Very useful video! also a simple quiz from a learn tail is a chef’s touch ❤ Thank you!
@ArjanCodes
@ArjanCodes Жыл бұрын
I'm glad you're enjoying Learntail :)
@danilshein4612
@danilshein4612 Жыл бұрын
Nice video as always! And examples are shown is pretty useful as long as your API is such simple one. In my case almost every API endpoint contains quite complicated business logic over the data that are gathered with several SQL requests that uses some results from previous ones. All kind of guides I've read about testing an API is always playing around kind of 'hello world' examples. The second issue for me is that some complex API endpoints requires DB filled with pretty complex set of data. Minimal viable data set is about 6 to 10 GB on disk, besides that it is additional effort to create it from production DB contents and ensure it is consistent.
@nathnaeldereje5127
@nathnaeldereje5127 2 ай бұрын
This is such a quality content. Thanks a lot. One question I have though is if we're using a layered architecture... say DDD with layers like the infrustructure, domain (usecases) and such a thing how do we test them independent of the other layer?
@devincunningham9717
@devincunningham9717 Жыл бұрын
if you have multiple assertions in one test that are not contingent on each other, I would recommend either splitting up the test case (sometimes a bit much) or using a package like pytest_check, which allows pytest to handle multiple checks without ending a test as soon as the first assertion fails. that way, you can get a more detailed report of everything that fails
@isaacomolewa9843
@isaacomolewa9843 10 ай бұрын
Is that not what the -vv flat does with pytest?
@jakubzbroda
@jakubzbroda Жыл бұрын
Exactly the problem I was facing! Thank you, Arjan.
@ArjanCodes
@ArjanCodes Жыл бұрын
Glad the video helped!
@oy12la
@oy12la Жыл бұрын
Very structured and helpful video giving some new aspects I was unsure about how to handle before. Many thanks! :) Was creating a FastAPI Project these days again for a first pitch of a small - specific - control module in a radio station. I was happy not having the need of using a database this time. In the project before (with DB) some it felt kind of unsatisfactory, as I had the feeling of writing some stuff two times (for API and for DB). Also I was not happy about the binding: The will of decoupling was strong in me, but I failed. Can't wait for refactoring this
@sammathew535
@sammathew535 Жыл бұрын
There was small discussion that we had on Coding Crashkurse channel on this topic. I think the approach to have dummy DBs (using say, SQLlite) may not be the best approach for doing Unit-tests for APIs or utilities. We should rather test with mock functions that can be checked for how often they've been called! Testing with DB might fit well with an integration test - say by the QA environment.
@carlosdavila2831
@carlosdavila2831 Жыл бұрын
I have a fastapi built on top of a Snowflake db (snowflake connector), which gives the team headaches to test because we are using Snowflake particular types like OBJECT in our db model, so we cannot reproduce it in a sqllite db for testing. Any suggestion?
@TheAremoh
@TheAremoh Жыл бұрын
Very helpful. Thank you
@ArjanCodes
@ArjanCodes Жыл бұрын
Glad the video helped!
@jamesclarke7259
@jamesclarke7259 Жыл бұрын
Amazing, I always learn so much. Thank you
@ArjanCodes
@ArjanCodes Жыл бұрын
Thank you for the kind words, James!
@agb2557
@agb2557 Жыл бұрын
Does pytest auto-recognise the setup and teardown functions? If so does it call them before and after each test, or before and after each test session?
@MatheusHenrique-rt9fc
@MatheusHenrique-rt9fc 3 ай бұрын
I have the same doubt
@plato4ek
@plato4ek 11 ай бұрын
9:00 Shouldn't the expression on the 48th line be inside the try-finally block? Seems like the `yield database` on its own cannot throw an exception.
@leovroy
@leovroy Жыл бұрын
Just what I was looking for. Thanks 👍🏻
@ArjanCodes
@ArjanCodes Жыл бұрын
Glad it was helpful!
@DeKeL_BaYaZi
@DeKeL_BaYaZi 7 ай бұрын
Hi Arjan, this video it amazing (I already commented the other day), but I can't find the test_operation file in your git repository
@FolkOverplay
@FolkOverplay Жыл бұрын
At 4:35 on line 72, you only close the DB connection if you don’t raise an error. Could that not creat problems with many sessions if the application is long-running?
@generalsnow8283
@generalsnow8283 Жыл бұрын
Where are the setup and teardown methods called from?
@MatheusHenrique-rt9fc
@MatheusHenrique-rt9fc 3 ай бұрын
I have the same doubt
@HarrisBallis
@HarrisBallis 6 ай бұрын
A video about testcontainers would be very nice
@bigoper
@bigoper 2 ай бұрын
Flawless ❤
@eladaharon1577
@eladaharon1577 10 ай бұрын
Great video, thanks for sharing that! 1 major thing about converting different layers of models: `return Item(**db_item.__dict__)` might throw `TypeError: API.__init__() got an unexpected keyword argument 'hash_value' ` for adding another field in DB: ``` @dataclass class DB: id: str name: str # adding hash_value to DB, not meant to be shared with API hash_value: bytearray @dataclass class API: id: str name: str d = DB(id="1", name="db", hash_value=bytearray([125, 125])) api = API(**d.__dict__) ``` tests will find this bug before production, but for that you need good coverage. I prefer having it explicitly done with a convertor function i.e: create_api(db) even if it's boring as hell. What do you think about c'tors receiving unpacked arguments with **?
@MatheusHenrique-rt9fc
@MatheusHenrique-rt9fc 3 ай бұрын
the "setup" and "teardown" doesn't ran automatically here. I had to put that on the test function. It is possible to run it automatically like in your code? Without fixtures?
@jccorman5848
@jccorman5848 Жыл бұрын
Is starting a db session for each route something that is done when using relational databases? I only know how to use MongoDB and a connection to the db is done once on startup.
@holgerbirne1845
@holgerbirne1845 10 ай бұрын
Thank you for the video! Where did you get this cool T-shirt from? :)
@emergency6300
@emergency6300 Жыл бұрын
@ArianCodes could you do a video doing the same but with unittest instead?
@lunalect
@lunalect Жыл бұрын
Nice vid. Are you interested in doing a similar video from a data engineering perspective, let’s say using polars and duckdb? Should be fun an probably a little more straightforward.
@Asgallu
@Asgallu Жыл бұрын
Could I use unittest instead of pytest with the code?
@christopherkiessling
@christopherkiessling 9 ай бұрын
I don't see why the need of creating a custom not found error, why don't just let the operation return none so the API part can handle it and raise the proper exception? That would be easier with the same separation of concerns. Great video BTW most tutorials don't address good practices as you do, appreciated.
@danielhernanbautistamantil1435
@danielhernanbautistamantil1435 Жыл бұрын
thank you
@ArjanCodes
@ArjanCodes Жыл бұрын
Glad you enjoyed it, Daniel!
@InvestorKiddd
@InvestorKiddd 8 ай бұрын
Beautiful
@ArjanCodes
@ArjanCodes 8 ай бұрын
Thank you!
@kristjanjonsson3843
@kristjanjonsson3843 Жыл бұрын
What if your api needs to talk to external things, like third party apis?
@baldpolnareff7224
@baldpolnareff7224 Жыл бұрын
I’m very curious how you would approach this using SQLModel, a new ORM made by the fastAPI author and built on top of fastAPI itself and SQLAlchemy
@PaolaVargas-k8r
@PaolaVargas-k8r Жыл бұрын
@ArjanCodes do you offer a mentorship service?
@ArjanCodes
@ArjanCodes Жыл бұрын
Hi! I currently don't offer any type of mentorship service, for the time being. However, my Software Designer Mindset course offers two different services that might interest you. When enrolling, you'll get access to my private student community on Discord, which I interact with quite often. Depending on the package you choose, you might also get a one-time custom code review by me in either Python, JavaScript, or Typescript! You can check my website for more information if you wish! :) Hope this comment helped.
@MuzafferCikay
@MuzafferCikay Жыл бұрын
You can group the functions that do the the database operation into a class which will be repository pattern. I would also like to see how to set up the deepest model objects creation. For instance one need to write test for a project management tool like Jira. If we want to test commet. We need to create project, team, member, status, task and finally a comment. Could you make a tutorial for deepest hieararchy model object creation?
@MuzafferCikay
@MuzafferCikay Жыл бұрын
It would also be good dealing with user authentication when sending API requests.
@MuzafferCikay
@MuzafferCikay Жыл бұрын
By the way the the rest o the application other than repositories classes can decoupled completely from the orm by returning the DTO from repository class methods.
@ayoubelmhamdi7920
@ayoubelmhamdi7920 Жыл бұрын
i like writing code from scratch rather than showing a screen shot of code
@JonasDC2
@JonasDC2 Жыл бұрын
Hi Arjan, I see you're using Item(**db_item.__dict__). Is there a reason behind using __dict__ instead of dict()?
@timbrap4693
@timbrap4693 Жыл бұрын
Do you mean dict(**db_item) ? or db_item.dict() ? Because of my knowledge sqlalchemy does not have that method. **__dict__** is a dunder that represents class attributes as a dictionary. Unpacking with ** results in initializing the pydantic model with keyword arguments. In this situation, he could have added a response_model=Item to the decorator function of the route. Fastapi would have handled the rest.
@zbaktube
@zbaktube Жыл бұрын
Hi! I would not call it Integration test, I would call just "integrationy". As it is not tested against the same type of db as the life system, and also it is hosted a bit differently. About testing my api: In a next comment, after a sleep 🙂
@lukashk.1770
@lukashk.1770 8 ай бұрын
Well this is a good intro,.But in production grade app things are done quite differently. FastAPI is async framework and sadly almost all tutorials show how to use FAstAPI with sync python only. In real life FastAPI app calls to DB are async, DB manages a pool of connections and everything gets quite a bit more complex. It would be nice to see a tutorial on more realistic setup of FastAPI DB layer
@ordinarygg
@ordinarygg Жыл бұрын
I put a monitor while you watch monitor. 2x different mappers to same SQL table... just to update row Look: JSON -> model.Pydantic -> serialize in dict -> serialize in SQLAlchemy ORM mapper -> serializes in dict for query -> sql query. Instead could be Pydantic -> SQL query or SQLAlchemy Core.
@random_act
@random_act 8 ай бұрын
what about a video regarding Testcontainers 😜
@udaym4204
@udaym4204 7 ай бұрын
please make video on to use sqlalchemy 2.0 with fatapi
@buchi8449
@buchi8449 Жыл бұрын
To be transparent, I am a bit Java-biased as I use both Java and Python daily. The solution explained in this video is not clean enough because you inject DB sessions into routers. But what the routers depend on are not DB sessions. The routers depend on operations. DB sessions are just internal details of the operations. From my Java-biased point of view, defining and injecting service objects providing operations is a cleaner solution. routers -> service objects -> DB sessions In this approach, we don't need in-memory SQLite for testing the routers. What we need is a mock of the service class. This approach makes tests of routers independent of DB access logic. By the way, what this video illustrates is something we have seen repeatedly in various programming languages and frameworks - fat controllers. And the mitigation for this problem is also always the same - the classic MVC model from the 1980s'.
@TheRich107
@TheRich107 Жыл бұрын
I agree with your approach however I think there is value is including the db in the scope of the controller tests. The python type system can mess up decimals and float types and it only surfaces when passed to the database. The value comes from checking that when that data has come from JSON into the controller and translated by the service, it might not be as you expect. I have had a type of decimal accept floats from the controller just fine. Individual unit tests passed but actually writing controller sourced data from the controller to the database through the service failed. In that case test would pass with TCP connection but socket failed.
@buchi8449
@buchi8449 Жыл бұрын
@@TheRich107 I understand your point, but I prefer having a clear separation of concerns. The ambiguity in type handling is a real pitfall in Python. But in this example of Web application tests, we should ensure strict type handling at the controller level (this is why we use Pydantic) and in assertions and mocks used in unit tests. The controller should focus on interfacing between the external world (Web clients) and business logic, and the business logic should be located somewhere else, like a service class. The injection of DB sessions may work in small and independent Web services. But generally, a Web service can rely on multiple databases and other external dependencies, including microservices in a company, files in S3 buckets and various services in cloud platforms. A DB session is just one of many dependencies maintaining application status. In our team, we encourage members to use predefined FastAPI project templates that include skeletons of routers and service classes with unit tests. Our templates aim to enforce a clear separation between controllers and models (services) from the beginning of every project. The overhead for introducing a service layer is literally zero. But its benefits are a lot.
@rolandobloom22
@rolandobloom22 Жыл бұрын
@@buchi8449 Do you have an example repository with that structure? I'm not referring to your company's repo, but perhaps you've come across something very similar to what you're discussing in public repositories?
@TheRich107
@TheRich107 Жыл бұрын
Yeah I like having a separate service layer certainly makes for easier testing. I get your point not sure I follow on the multi dependency. Let me check my dependency, It makes sense that I don't have dependency injections for S3 buckets in the controller and that is Purley a service layer concern. I might have to play around with doing the same for the database. The fast api author does know what he is doing. I would be interested to hear the pros of having the DB injected into the controller and why that is the documented way to use the framework. When I have instantiated a database connection at the service layer before I didn't find a way to override it in the tests. (I have several DB sessions running side by side in the tests)
@TheRich107
@TheRich107 4 ай бұрын
​@maihuynhtuanvu2254 Not got sample code, was production code 😅 good old sentry had my back though.
@Mjuziks
@Mjuziks Жыл бұрын
👌
@garrywreck4291
@garrywreck4291 Жыл бұрын
IMHO HTTP handlers should know nothing about DB! There should be some "service" entity which you inject into handlers. Then you test HTTP-related stuff with a fake service. The "service" itself is tested separately and it knows nothing about HTTP. My personal favorite way is to have a class like "Application" which encapsulates HTTP app (e.g. Flask, FastAPI, etc.) and some service (e.g. Service class). The Application accepts the service via initializer. HTTP handlers have access to the service. Then you init the "Application" in tests passing it some FakeService, initialize a test HTTP client, and test your requests.
@codingcrashkurse6429
@codingcrashkurse6429 Жыл бұрын
That´s what I also say. Unittests are there to test a code unit. Unit tests should not be there for integration tests
@buchi8449
@buchi8449 Жыл бұрын
I made the same comment. Injecting DB sessions to routers is not clean. We should inject service objects into routers and use mocks of the service objects to test the routers. It is not difficult to modify the current code in the video to have a service class. In the video, the "operations" are extracted as pure functions. However, this is not a good model since the operations have side effects on the state of the application. The better approach is extracting the "operations" as methods of a service class. Then we can inject DB sessions to __init__ of the service class, and inject instances of the service class into routers (FastAPI's DI can recursively inject dependencies).
@nelsonmacy1010
@nelsonmacy1010 3 ай бұрын
So much glue code and scaffolding. Maybe you should learn SQL. Reminds me of people who rely on Microsoft GUI vs know how to use bash.
@nicholasobrien5914
@nicholasobrien5914 Жыл бұрын
Hey man are you on the Carnivore Diet ? You look like you lost some weight!
@eli1882
@eli1882 Жыл бұрын
Unsurprisingly, a yotuber without a successful programming career has 0 skills in software architecture. You're helping some, but you disregard talking about some of the more important decisions you must make before you even start typing.
How To Write Unit Tests For Existing Python Code // Part 1 of 2
25:07
Avoid These BAD Practices in Python OOP
24:42
ArjanCodes
Рет қаралды 81 М.
VIP ACCESS
00:47
Natan por Aí
Рет қаралды 30 МЛН
Cat mode and a glass of water #family #humor #fun
00:22
Kotiki_Z
Рет қаралды 42 МЛН
Why You NEED To Learn FastAPI | Hands On Project
21:15
Travis Media
Рет қаралды 172 М.
Requests vs HTTPX vs Aiohttp
15:11
ArjanCodes
Рет қаралды 40 М.
Protocol Or ABC In Python - When to Use Which One?
23:45
ArjanCodes
Рет қаралды 206 М.
COMPLETE No-Nonsense VSCode Setup for Python Devs
26:05
ArjanCodes
Рет қаралды 50 М.
How to Use FastAPI: A Detailed Python Tutorial
20:38
ArjanCodes
Рет қаралды 271 М.
SQLAlchemy Makes Using Python Databases EASY
19:08
Eric Roby
Рет қаралды 7 М.
Dependency Injection, The Best Pattern
13:16
CodeAesthetic
Рет қаралды 904 М.
Unit Testing in Spring Boot with JUnit 5 and Mockito | Part 1
30:37
Python Decorators: The Complete Guide
27:59
ArjanCodes
Рет қаралды 162 М.