from fastlite import *
from fastcore.utils import *
from fastcore.net import urlsave
fastlite
fastlite
provides some little quality-of-life improvements for interactive use of the wonderful sqlite-utils library. It’s likely to be particularly of interest to folks using Jupyter.
Install
pip install fastlite
Overview
We demonstrate fastlite
‘s features here using the ’chinook’ sample database.
= 'https://github.com/lerocha/chinook-database/raw/master/ChinookDatabase/DataSources/Chinook_Sqlite.sqlite'
url = Path('chinook.sqlite')
path if not path.exists(): urlsave(url, path)
= database("chinook.sqlite") db
Databases have a t
property that lists all tables:
= db.t
dt dt
Album, Artist, Customer, Employee, Genre, Invoice, InvoiceLine, MediaType, Playlist, PlaylistTrack, Track
You can use this to grab a single table…:
= dt.artists
artist artist
<Table artists (does not exist yet)>
= dt.Artist
artist artist
<Table Artist (ArtistId, Name)>
…or multiple tables at once:
'Artist','Album','Track','Genre','MediaType'] dt[
[<Table Artist (ArtistId, Name)>,
<Table Album (AlbumId, Title, ArtistId)>,
<Table Track (TrackId, Name, AlbumId, MediaTypeId, GenreId, Composer, Milliseconds, Bytes, UnitPrice)>,
<Table Genre (GenreId, Name)>,
<Table MediaType (MediaTypeId, Name)>]
It also provides auto-complete in Jupyter, IPython, and nearly any other interactive Python environment:
You can check if a table is in the database already:
'Artist' in dt
True
Column work in a similar way to tables, using the c
property:
= artist.c
ac ac
ArtistId, Name
Auto-complete works for columns too:
Columns, tables, and view stringify in a format suitable for including in SQL statements. That means you can use auto-complete in f-strings.
= f"select * from {artist} where {ac.Name} like 'AC/%'"
qry print(qry)
select * from "Artist" where "Artist"."Name" like 'AC/%'
You can view the results of a select query using q
:
db.q(qry)
[{'ArtistId': 1, 'Name': 'AC/DC'}]
Views can be accessed through the v
property:
= dt.Album
album
= f"""select {album}.*
acca_sql from {album} join {artist} using (ArtistId)
where {ac.Name} like 'AC/%'"""
"AccaDaccaAlbums", acca_sql, replace=True)
db.create_view(= db.q(f"select * from {db.v.AccaDaccaAlbums}")
acca_dacca acca_dacca
[{'AlbumId': 1,
'Title': 'For Those About To Rock We Salute You',
'ArtistId': 1},
{'AlbumId': 4, 'Title': 'Let There Be Rock', 'ArtistId': 1}]
Dataclass support
A dataclass
type with the names, types, and defaults of the tables is created using dataclass()
:
= album.dataclass() album_dc
Let’s try it:
= album_dc(**acca_dacca[0])
album_obj album_obj
Album(AlbumId=1, Title='For Those About To Rock We Salute You', ArtistId=1)
You can get the definition of the dataclass using fastcore’s dataclass_src
– everything is treated as nullable, in order to handle auto-generated database values:
= dataclass_src(album_dc)
src 'python') hl_md(src,
@dataclass
class Album:
int | None = None
AlbumId: str | None = None
Title: int | None = None ArtistId:
Because dataclass()
is dynamic, you won’t get auto-complete in editors like vscode – it’ll only work in dynamic environments like Jupyter and IPython. For editor support, you can export the full set of dataclasses to a module, which you can then import from:
'db_dc') create_mod(db,
from db_dc import Track
Track()
Track(TrackId=None, Name=None, AlbumId=None, MediaTypeId=None, GenreId=None, Composer=None, Milliseconds=None, Bytes=None, UnitPrice=None)
Indexing into a table does a query on primary key:
1] dt.Track[
Track(TrackId=1, Name='For Those About To Rock (We Salute You)', AlbumId=1, MediaTypeId=1, GenreId=1, Composer='Angus Young, Malcolm Young, Brian Johnson', Milliseconds=343719, Bytes=11170334, UnitPrice=0.99)
There’s a shortcut to select from a table – just call it as a function. If you’ve previously called dataclass()
, returned iterms will be constructed using that class by default. There’s lots of params you can check out, such as limit
:
=2) album(limit
[Album(AlbumId=1, Title='For Those About To Rock We Salute You', ArtistId=1),
Album(AlbumId=2, Title='Balls to the Wall', ArtistId=2)]
Pass a truthy value as with_pk
and you’ll get tuples of primary keys and records:
=1, limit=2) album(with_pk
[(1,
Album(AlbumId=1, Title='For Those About To Rock We Salute You', ArtistId=1)),
(2, Album(AlbumId=2, Title='Balls to the Wall', ArtistId=2))]
Indexing also uses the dataclass by default:
5] album[
Album(AlbumId=5, Title='Big Ones', ArtistId=3)
If you set xtra
fields, then indexing is also filtered by those. As a result, for instance in this case, nothing is returned since album 5 is not created by artist 1:
=1)
album.xtra(ArtistId
try: album[5]
except NotFoundError: print("Not found")
Not found
The same filtering is done when using the table as a callable:
album()
[Album(AlbumId=1, Title='For Those About To Rock We Salute You', ArtistId=1),
Album(AlbumId=4, Title='Let There Be Rock', ArtistId=1)]
Core design
The following methods accept **kwargs
, passing them along to the first dict
param:
create
transform
transform_sql
update
insert
upsert
lookup
We can access a table that doesn’t actually exist yet:
= dt.cats
cats cats
<Table cats (does not exist yet)>
We can use keyword arguments to now create that table:
id=int, name=str, weight=float, uid=int, pk='id')
cats.create('sql') hl_md(cats.schema,
CREATE TABLE [cats] (
id] INTEGER PRIMARY KEY,
[
[name] TEXT,FLOAT,
[weight] uid] INTEGER
[ )
It we set xtra
then the additional fields are used for insert
, update
, and delete
:
=2)
cats.xtra(uid= cats.insert(name='meow', weight=6) cat
The inserted row is returned, including the xtra ‘uid’ field.
cat
{'id': 1, 'name': 'meow', 'weight': 6.0, 'uid': 2}
Using **
in update
here doesn’t actually achieve anything, since we can just pass a dict
directly – it’s just to show that it works:
'name'] = "moo"
cat['uid'] = 1
cat[**cat)
cats.update( cats()
[{'id': 1, 'name': 'moo', 'weight': 6.0, 'uid': 2}]
Attempts to update or insert with xtra fields are ignored.
An error is raised if there’s an attempt to update a record not matching xtra
fields:
=1)
cats.xtra(uidtry: cats.update(**cat)
except NotFoundError: print("Not found")
Not found
This all also works with dataclasses:
=2)
cats.xtra(uid
cats.dataclass()= cats[1]
cat cat
Cats(id=1, name='moo', weight=6.0, uid=2)
cats.drop() cats
<Table cats (does not exist yet)>
Alternatively, you can create a table from a class. If it’s not already a dataclass, it will be converted into one. In either case, the dataclass will be created (or modified) so that None
can be passed to any field (this is needed to support fields such as automatic row ids).
class Cat: id:int; name:str; weight:float; uid:int
= db.create(Cat) cats
'sql') hl_md(cats.schema,
CREATE TABLE [cat] (
id] INTEGER PRIMARY KEY,
[
[name] TEXT,FLOAT,
[weight] uid] INTEGER
[ )
= Cat(name='咪咪', weight=9)
cat cats.insert(cat)
Cat(id=1, name='咪咪', weight=9.0, uid=None)
cats.drop()
Manipulating data
We try to make the following methods as flexible as possible. Wherever possible, they support Python dictionaries, dataclasses, and classes.
.insert()
Creates a record. In the name of flexibility, we test that dictionaries, dataclasses, and classes all work. Returns an instance of the updated record.
Insert using a dictionary.
'name': 'Rex', 'weight': 12.2}) cats.insert({
Cat(id=1, name='Rex', weight=12.2, uid=UNSET)
Insert using a dataclass.
= cats.dataclass()
CatDC ='Tom', weight=10.2)) cats.insert(CatDC(name
Cat(id=2, name='Tom', weight=10.2)
Insert using a standard Python class
= cats.insert(Cat(name='Jerry', weight=5.2)) cat
.update()
Updates a record using a Python dict, dataclasses, and classes all work and returns an instance of the updated record.
Updating from a Python dict:
dict(id=cat.id, name='Jerry', weight=6.2)) cats.update(
Cat(id=3, name='Jerry', weight=6.2)
Updating from a dataclass:
id=cat.id, name='Jerry', weight=6.3)) cats.update(CatDC(
Cat(id=3, name='Jerry', weight=6.3)
Updating using a class:
id=cat.id, name='Jerry', weight=5.7)) cats.update(Cat(
Cat(id=3, name='Jerry', weight=5.7)
.delete()
Removing data is done by providing the primary key value of the record.
# Farewell Jerry!
id) cats.delete(cat.
<Table cat (id, name, weight)>
Importing CSV/TSV/etc
Database.import_file
Database.import_file (table_name, file, format=None, pk=None)
Import path or handle file
to new table table_name
You can pass a file name, string, bytes, or open file handle to import_file
to import a CSV:
= Database(":memory:")
db = """id,name,age
csv_data 1,Alice,30
2,Bob,25
3,Charlie,35"""
= db.import_file("people", csv_data)
table table()
[{'id': 1, 'name': 'Alice', 'age': 30},
{'id': 2, 'name': 'Bob', 'age': 25},
{'id': 3, 'name': 'Charlie', 'age': 35}]
Diagrams
If you have graphviz installed, you can create database diagrams:
diagram(db.tables)
Pass a subset of tables to just diagram those. You can also adjust the size and aspect ratio.
'Artist','Album','Track','Genre','MediaType'], size=8, ratio=0.4) diagram(db.t[