All Projects → ThibTrip → Pangres

ThibTrip / Pangres

Licence: unlicense
SQL upsert using pandas DataFrames for PostgreSQL, SQlite and MySQL with extra features

Programming Languages

python
139335 projects - #7 most used programming language
python3
1442 projects

Projects that are alternatives of or similar to Pangres

Sqlcheck
Automatically identify anti-patterns in SQL queries
Stars: ✭ 2,062 (+2932.35%)
Mutual labels:  sql, postgresql, sqlite3
Shmig
Database migration tool written in BASH.
Stars: ✭ 408 (+500%)
Mutual labels:  sql, postgresql, sqlite3
Rom Sql
SQL support for rom-rb
Stars: ✭ 169 (+148.53%)
Mutual labels:  sql, postgresql, sqlite3
Purescript Selda
A type-safe, high-level SQL library for PureScript
Stars: ✭ 72 (+5.88%)
Mutual labels:  sql, postgresql, sqlite3
Lucid
AdonisJS official SQL ORM. Supports PostgreSQL, MySQL, MSSQL, Redshift, SQLite and many more
Stars: ✭ 613 (+801.47%)
Mutual labels:  sql, postgresql, sqlite3
Sworm
a write-only ORM for Node.js
Stars: ✭ 128 (+88.24%)
Mutual labels:  sql, postgresql, sqlite3
Walkable
A Clojure(script) SQL library for building APIs: Datomic® (GraphQL-ish) pull syntax, data driven configuration, dynamic filtering with relations in mind
Stars: ✭ 384 (+464.71%)
Mutual labels:  sql, postgresql, sqlite3
Electrocrud
Database CRUD Application Built on Electron | MySQL, Postgres, SQLite
Stars: ✭ 1,267 (+1763.24%)
Mutual labels:  sql, postgresql, sqlite3
Trdsql
CLI tool that can execute SQL queries on CSV, LTSV, JSON and TBLN. Can output to various formats.
Stars: ✭ 593 (+772.06%)
Mutual labels:  sql, postgresql, sqlite3
Qb
The database toolkit for go
Stars: ✭ 524 (+670.59%)
Mutual labels:  sql, postgresql, sqlite3
Choochoo
Training Diary
Stars: ✭ 186 (+173.53%)
Mutual labels:  pandas, postgresql, sqlite3
Vscode Sqltools
Database management for VSCode
Stars: ✭ 741 (+989.71%)
Mutual labels:  sql, postgresql, sqlite3
Knex
A query builder for PostgreSQL, MySQL, CockroachDB, SQL Server, SQLite3 and Oracle, designed to be flexible, portable, and fun to use.
Stars: ✭ 15,083 (+22080.88%)
Mutual labels:  sql, postgresql, sqlite3
Openrecord
Make ORMs great again!
Stars: ✭ 474 (+597.06%)
Mutual labels:  sql, postgresql, sqlite3
Pmacct
pmacct is a small set of multi-purpose passive network monitoring tools [NetFlow IPFIX sFlow libpcap BGP BMP RPKI IGP Streaming Telemetry].
Stars: ✭ 677 (+895.59%)
Mutual labels:  sql, postgresql, sqlite3
Usql
Universal command-line interface for SQL databases
Stars: ✭ 6,869 (+10001.47%)
Mutual labels:  sql, postgresql, sqlite3
Massive
A Module for Utilizing MassiveJS with NestJS
Stars: ✭ 27 (-60.29%)
Mutual labels:  sql, postgresql
Efcore.pg
Entity Framework Core provider for PostgreSQL
Stars: ✭ 838 (+1132.35%)
Mutual labels:  sql, postgresql
Node Pg Migrate
Node.js database migration management for Postgresql
Stars: ✭ 838 (+1132.35%)
Mutual labels:  sql, postgresql
Soci
Official repository of the SOCI - The C++ Database Access Library
Stars: ✭ 960 (+1311.76%)
Mutual labels:  postgresql, sqlite3

CircleCI codecov PyPI version

pangres

pangres logo

Thanks to freesvg.org for the logo assets

Upsert with pandas DataFrames (ON CONFLICT DO NOTHING or ON CONFLICT DO UPDATE) for PostgreSQL, MySQL, SQlite and potentially other databases behaving like SQlite (untested) with some additional optional features (see features). Upserting can be done with primary keys or unique keys. Pangres also handles the creation of non existing SQL tables and schemas.

Features

  1. (optional) Automatical column creation (when a column exists in the DataFrame but not in the SQL table).
  2. (optional) Automatical column type alteration for columns that are empty in the SQL table (except for SQlite where alteration is limited).
  3. Creates the table if it is missing.
  4. Creates missing schemas in Postgres (and potentially other databases that have a schema system).
  5. JSON is supported (with pd.to_sql it does not work) with some exceptions (see Gotchas and caveats).
  6. Fast (except for SQlite where some help is needed).
  7. Will work even if not all columns defined in the SQL table are there.
  8. SQL injection safe (schema, table and column names are escaped and values are given as parameters).

Tested with

  • Python 3.7.3 and Python 3.8.0
  • MySQL 5.7.29 using pymysql 0.9.3
  • PostgreSQL 9.6.17 using psycopg2 2.8.4
  • SQlite 3.28.0 using sqlite3 2.6.0

Gotchas and caveats

All flavors

  1. We can't create JSON columns automatically but we can insert JSON like objects (list, dict) in existing JSON columns.

Postgres

  1. "%", ")" and "(" in column names will most likely cause errors with PostgreSQL (this is due to psycopg2 and also affect pd.to_sql). Use the function pangres.fix_psycopg2_bad_cols to "clean" the columns in the DataFrame. You'll also have to rename columns in the SQL table accordingly (if the table already exists).
  2. Even though we only do data type alteration on empty columns, since we don't want to lose column information (e.g. constraints) we use true column alteration (instead of drop+create) so the old data type must be castable to the new data type. Postgres seems a bit restrictive in this regard even when the columns are empty (e.g. BOOLEAN to TIMESTAMP is impossible).

SQlite

  1. SQlite must be version 3.24.4 or higher! UPSERT syntax did not exist before.
  2. Column type alteration is not possible for SQlite.
  3. SQlite inserts can be at worst 5 times slower than pd.to_sql for some reasons. If you can help please contact me!
  4. Inserts with 1000 columns or more are not supported due to a restriction of 999 parameters per queries. One way to fix this would inserting the columns progressively but this seems quite tricky. If you know a better way please contact me.

MySQL

  1. MySQL will often change the order of the primary keys in the SQL table when using INSERT... ON CONFLICT.. DO NOTHING/UPDATE. This seems to be the expected behavior so nothing we can do about it but please mind that!
  2. You may need to provide SQL dtypes e.g. if you have a primary key with text you will need to provide a character length (e.g. VARCHAR(50)) because MySQL does not support indices/primary keys with flexible text length. pd.to_sql has the same issue.

Notes

This is a library I was using in production in private with very good results and decided to publish.

Ideally such features will be integrated into pandas since there is already a PR on the way) and I would like to give the option to add columns via another PR.

There is also pandabase which does almost the same thing (plus lots of extra features) but my implementation is different. Btw big thanks to pandabase and the sql part of pandas which helped a lot.

Installation

pip install pangres

Additionally depending on which database you want to work with you will need to install the corresponding library (note that SQlite is included in the standard library):

  • Postgres
pip install psycopg2
  • MySQL
pip install pymysql

Usage

Head over to pangres' wiki!

Contributing

Pull requests/issues are welcome.

Testing

You will need a SQlite, MySQL and Postgres database available for testing.

Clone pangres then set your curent working directory to the root of the cloned repository folder. Then use the commands below. You will have to replace the following variables in those commands:

  • SQLITE_CONNECTION_STRING: replace with a SQlite sqlalchemy connection string (e.g. "sqlite:///test.db")
  • POSTGRES_CONNECTION_STRING: replace with a Postgres sqlalchemy connection string (e.g. "postgres:///user:[email protected]:5432/database"). Specifying schema is optional for postgres (will default to public).
  • PG_SCHEMA (optional): schema for postgres (defaults to public)
  • MYSQL_CONNECTION_STRING: replace with a MySQL sqlalchemy connection string (e.g. "mysql+pymysql:///user:[email protected]:3306/database")
# 1. Create and activate the build environment
conda env create -f environment.yml
conda activate pangres-dev
# 2. Install pangres in editable mode (changes are reflected upon reimporting)
pip install -e .
# 3. Run pytest
# -s prints stdout
# -v prints test parameters
# --cov=./pangres shows coverage only for pangres
# --doctest-modules tests with doctest in all modules
pytest -s -v pangres --cov=pangres --doctest-modules --sqlite_conn=$SQLITE_CONNECTION_STRING --pg_conn=$POSTGRES_CONNECTION_STRING --mysql_conn=$MYSQL_CONNECTION_STRING --pg_schema=tests
Note that the project description data, including the texts, logos, images, and/or trademarks, for each open source project belongs to its rightful owner. If you wish to add or remove any projects, please contact us at [email protected].