Movatterモバイル変換


[0]ホーム

URL:


Skip to content
DEV Community
Log in Create account

DEV Community

Cover image for Deploy FastAPI application with SQLite on Fly.io
vladkens
vladkens

Posted on

     

Deploy FastAPI application with SQLite on Fly.io

Cloud solutions are good for medium and large projects, but too heavy for small personal projects. If you want to launch something small (a few api endpoins and a small repository), there are three options:

  • Use the same approaches as for "big" projects (AWS ECS/EKS, RDS), but they are redundant, and infrastructure code can be larger than code of the actual project. Also it's expensive (~$100).
  • Use serverless solutions (Lambda, Vercel). Most cloud providers have such solutions, but these services have difficulties with simple databases – they provide cheap vendor solutions (AWS) or require a managed database, which again is expensive (mostly nothing for serverless, ~$20 for DB)
  • Use VPS with Docker. It is cheap (~$5 for small machine) and almost no need to manage infrastructure, but deployments sucks (needs private or self-hosted regestry, SSH access from CI).

I usually write my small applications using SQLite, it's a handy little single file database that works in any programming language and can be copied to local machine to analyze data for example. So I was looking for some middleware solution that combines the serveless approach, ease of deployment and ability to use SQLite and found Fly.io.

Setup

If you don't have an account in Fly.io – you need tocreate it. AlsoCLI tool calledflyctl required to manage projects. Fly.io can be deployed both locally and from CI.

flyctl makes deploy from project's root folder from Dockerfile, which is cool, because same Dockerfile can be used in other systems. For play with Fly.io, I prepared a simple FastAPI project that stores state in database – generic url shortener with click counting.

Dockerfile:

FROM python:3.13-alpineWORKDIR /appCOPY ./requirements.txt .RUNpipinstall--no-cache-dir--upgrade-r requirements.txtCOPY . /appENV HOST=0.0.0.0 PORT=8080EXPOSE ${PORT}CMD uvicorn main:app --host ${HOST} --port ${PORT}
Enter fullscreen modeExit fullscreen mode

main.py:

importasyncioimportrandomimportstringfromurllib.parseimporturlparseimportaiosqlitefromfastapiimportFastAPI,HTTPException,Requestfromfastapi.responsesimportRedirectResponseDB_PATH="/data/app.db"app=FastAPI()asyncdefget_db()->aiosqlite.Connection:ifdb:=getattr(get_db,"_db",None):ifdb.is_alive:returndbdb=awaitaiosqlite.connect(DB_PATH,loop=asyncio.get_event_loop())db.row_factory=aiosqlite.Rowqs="""    CREATE TABLE IF NOT EXISTS links (        created_at INTEGER DEFAULT (strftime('%s','now')),        short_code TEXT PRIMARY KEY,        full_url TEXT NOT NULL,        clicks INTEGER DEFAULT 0    )"""awaitdb.execute(qs)awaitdb.commit()setattr(get_db,"_db",db)returndbdefrandom_code(length=8)->str:alphabet=string.ascii_letters+string.digitsreturn"".join(random.choice(alphabet)forxinrange(length))defis_valid_url(url:str)->bool:try:parts=urlparse(url)returnall([parts.scheme,parts.netloc])exceptValueError:returnFalse@app.post("/")asyncdefshorten(url:str,req:Request):ifnotis_valid_url(url):raiseHTTPException(status_code=400,detail="Invalid URL")host=req.headers.get("host")ifhostisNone:raiseHTTPException(status_code=500,detail="Missing host header")short_code=random_code()db=awaitget_db()qs="INSERT INTO links (short_code, full_url) VALUES (?, ?)"awaitdb.execute(qs,(short_code,url))awaitdb.commit()returnf"https://{host}/{short_code}"@app.get("/")asyncdeflist_links():db=awaitget_db()qs="SELECT short_code, full_url, clicks FROM links ORDER BY created_at DESC"asyncwithdb.execute(qs)ascursor:returnawaitcursor.fetchall()@app.get("/{short_code}")asyncdefredirect(short_code:str):db=awaitget_db()qs="""    UPDATE links SET clicks = clicks + 1 WHERE short_code = ?    RETURNING full_url"""asyncwithdb.execute(qs,(short_code,))ascursor:ifrow:=awaitcursor.fetchone():returnRedirectResponse(row["full_url"])raiseHTTPException(status_code=404)
Enter fullscreen modeExit fullscreen mode

requirements.txt:

aiosqlitefastapiuvicorn
Enter fullscreen modeExit fullscreen mode

Deploy

To deploy our code, first we need to create a Fly.io project. This can be done either in the web interface or withflyctl. To create proejct with CLU tool in root folder (where code located) flyctl launch should be runned. This command will offer to select desired hardware and will createfly.toml file:

fly launch--build-only
Enter fullscreen modeExit fullscreen mode

You can modify project in future by changing parameters in this file or via web ui. The basicfly.toml looks fine, but SQLite requires Storage, which can be created with:

fly volumes create sqlite_data-s 1-r ams
Enter fullscreen modeExit fullscreen mode

where-s 1 sets volume size to 1 GB (default is 3 GB), and-r is region in which volume will be created (use same region in which Fly.io project is created). You can always change storage size later.

The last thing to do is to add amounts section tofly.toml, which attaches the volume to the application:

[mounts]source="sqlite_data"destination="/data"
Enter fullscreen modeExit fullscreen mode

sqlite_data is the name of the storage,/data is the path where volume will be connected. This is essentially same asdocker run --mount source=sqlite_data,target=/data or corresponding Docker Compose section.

SQLite cannot be writable from more than one app, and Fly.io by default creates 2 instances for an app, so we can specify the number of replicas as one just in case:

fly scale count 1
Enter fullscreen modeExit fullscreen mode

All configurations are done now and we can deploy our app with command:

fly deploy
Enter fullscreen modeExit fullscreen mode

The app should boot successfully and the public DNS name will be printed to console. Now we can check it out by posting some url to shortener:

❯ curl-X POST'https://fly-fastapi-sqlite.fly.dev/?url=https://example.com'https://fly-fastapi-sqlite.fly.dev/8ydmfAcK
Enter fullscreen modeExit fullscreen mode

Then we can visit this link, it should redirect tohttps://example.com. Finally, we can check that clicks are updated:

❯ curl-s'https://fly-fastapi-sqlite.fly.dev/' | jq[{"short_code":"8ydmfAcK","full_url":"https://example.com","clicks": 1}]
Enter fullscreen modeExit fullscreen mode

To check that database state saved between deployments, we can perform new deployment withfly deploy and check that links list remained same as above (1 link, 1 click).

Migrations

If you are using an external solution for migrations, rather than running them from code when app starts, then only way to run migration is to put it in Dockerfile as part of the RUN command.

Backup

We can connect to machine withfly ssh console and then in/data folder interact with database file. Also we can copy database file to local machine with:

fly ssh sftp get /data/app.db ./app-backup.db
Enter fullscreen modeExit fullscreen mode

Conclusion

Fly.io is a simple and convenient service for deploying applications. Deploy works from Docker Containers, additional services include PSQL, Redis, S3 like storage (unlike Vercel). It's cheap, the cheapest service costs 3 dollars (1 shared CPU / 256 MB) – maybe even less, if you have little traffic – container shuts down after a few minutes without activity and automatically turns on when traffic appears.

On downside, there is no built-in solution for scheduled tasks – instead, the official solution is to set up a separate server withcrontab and run tasks from it – it's kind of creepy.

Top comments(0)

Subscribe
pic
Create template

Templates let you quickly answer FAQs or store snippets for re-use.

Dismiss

Are you sure you want to hide this comment? It will become hidden in your post, but will still be visible via the comment'spermalink.

For further actions, you may consider blocking this person and/orreporting abuse

  • Joined

More fromvladkens

DEV Community

We're a place where coders share, stay up-to-date and grow their careers.

Log in Create account

[8]ページ先頭

©2009-2025 Movatter.jp