Build your e-commerce ea(a)sily
Project description
Environemnt As A Software
Build an environment with a database and a REST API.
Requirements
- Python 3.12
- PostgreSQL 13.4
- Redis 6.2.6 (optional but recommended)
Initial setup
Create a database in PostgreSQL:
create database database_name;
Create a virtual environment and install the dependencies:
python -m venv venv
source venv/bin/activate
pip install --upgrade pip
pip install -r requirements.txt
Set the environment variables:
POSTGRES_URI=<SQL_DATABASE_URI>
# macOS only
OBJC_DISABLE_INITIALIZE_FORK_SAFETY=YES
# Optional for Redis
REDIS_URI=<REDIS_URI>
Initialize alembic:
# alembic_init.py
from eaasy.extensions.migration import init
from argparse import ArgumentParser, Namespace as ArgumentNamespace
def get_arguments() -> ArgumentNamespace:
parser = ArgumentParser(description='Alembic migration helper')
parser.add_argument('sql_url', metavar='sql_url', type=str, help='SQLAlchemy URL')
parser.add_argument('--path', '-p', metavar='path', type=str, help='Alembic path', default='src/alembic')
parser.add_argument('--tables-folder', '-t', metavar='tables_folder', type=str, help='Tables folder', default='src/tables')
args = parser.parse_args()
if not args.sql_url:
raise Exception('SQLAlchemy URL is required')
return args
if __name__ == '__main__':
args = get_arguments()
init(args.sql_url, args.path, args.tables_folder)
Launch the script to build the alembic folder:
python alembic_init.py <SQL_DATABASE_URI> # --path src/alembic (optional)
Create a table:
# src/tables/user.py
from eaasy import BaseEntity, Audit
from sqlalchemy import Column, String
class UserProperties:
firstName = Column(String, nullable=False)
lastName = Column(String, nullable=False)
email = Column(String, nullable=False, unique=True)
class User(BaseEntity, UserProperties, Audit):
__tablename__ = 'users'
And add it to the src/tables/__init__.py file:
# src/tables/__init__.py
from .user import User
__all__ = ['User']
Run the migration:
alembic revision --autogenerate -m "Create users table"
alembic upgrade head
Run the application
Create a main module:
# app.py
from eaasy import Eaasy, GunEaasy
from eaasy.extensions import build_model, build_resource
from src.tables.user import User, UserProperties
api = Eaasy(
name=__name__,
title='API',
version='1.0',
description='A simple API',
doc='/swagger'
)
# Create models for User resource (GET, POST and PUT)
user_ns, get_model = build_model(User)
user_ns, upsert_model = build_model(UserProperties, namespace=user_ns)
# Build and register resource
build_resource(User, user_ns, get_model, upsert_model)
# # Add namespace to API
api.add_namespace(user_ns)
app = api.get_app() # Required if you want to perform flask operations
if __name__ == '__main__':
options = {
'bind': '%s:%s' % ('0.0.0.0', '8080'),
'workers': 1
}
GunEaasy(app, options).run()
Run the application:
python app.py
# or
gunicorn app:app
# or
flask run
Features
Custom endpoints
By default the build_resource method build a resource with these enabled endpoints:
get_all-> GET /entity/ # Get all entitiespost-> POST /entity/ # Create new entityget_by_id-> GET /entity/id:int # Get entity by idput-> PUT /entity/id:int # Edit entity by iddelete-> DELETE /entity/id:int # Delete entity by id
You can disable one or more endpoints by setting to False the correspoing key, for instance:
# Build resource without get_by_id endpoint
build_resource(User, user_ns, get_model, upsert_model, get_by_id=False)
File exports
Enable an Excel download by adding the file_export flag when building the resource:
build_resource(
User,
user_ns,
get_model,
upsert_model,
file_export=True,
file_headers=['id', 'firstName', 'lastName', 'email'],
file_name='users.xlsx'
)
GET /entity/file/streams an.xlsxattachment with every entity returned byget_all.file_headersis optional. When omitted, columns are inferred from the collected entities.file_nameis optional. The filename defaults to<EntityName>.xlsx.
Excel generation uses openpyxl.
File imports
You can let clients upload spreadsheet data and upsert rows by enabling file_import:
build_resource(
User,
user_ns,
get_model,
upsert_model,
file_import=True,
file_preview=True,
file_unique_fields=['email'], # Columns used to detect duplicates
file_field='file', # Optional form field name
file_sheet='Sheet1' # Optional sheet override
)
POST /entity/file/accepts a multipart form upload containing an.xlsxfile.- Rows are converted into dictionaries using the header row.
- If
file_unique_fieldsis provided (or defaults to['id']when the column exists), matching rows are updated while new rows are created. - Multiple columns are supported in
file_unique_fields(e.g.['name', 'category']) and must uniquely identify a row. - The endpoint responds with the number of
createdandupdatedrecords. POST /entity/file/preview/accepts the same payload and returns the parsed headers plus rows in JSON for a pre-import review.
Spreadsheet parsing also relies on openpyxl; ensure the dependency is installed before enabling the feature.
Callbacks
You can add callbacks to the resources:
def after_post(data):
print(data.firstName) # 'data' represent the model of the object created after the POST request
build_resource(User, user_ns, get_model, upsert_model, on_post=after_post)
Available callbacks:
on_poston_puton_delete
Limit rate
API requests can be limited by the number of requests in an interval of time.
NOTE: the application should be running with Redis properly configured. See environment variables in Initial setup section.
You can enable limiter using enable_limiter parameters:
api = Eaasy(
name=__name__,
title='API',
version='1.0',
description='A simple API',
doc='/swagger',
enable_limiter=True
)
And set the limit for all methods:
build_resource(User, user_ns, get_model, upsert_model, limit='5 per minute')
Or specify a limit for each method:
build_resource(User, user_ns, get_model, upsert_model, get_all_limit='5 per minute')
Available arguments:
get_all_limitget_by_id_limitpost_limitput_limitdelete_limit
Logger
By providing logger parameter in Eaasy class you can:
api = Eaasy(
logger=True # Configure default logger
)
api = Eaasy(
logger=custom_logger # Or provide your own logger
)
And you can also extract and wherever you want (especially if you want to use the default one):
logger = api.logger # This raises an exception if not configured
OpenIDConnect (from flask_oidc)
Same for the OpenIdConnet instance:
api = Eaasy(
oidc=True # Configure default OIDC
)
api = Eaasy(
oidc=OpenIDConnect(app) # Or provide your own OIDC (OpenIDConnect from flask_oidc package only)
)
Extract and use it:
oidc = api.oidc # This raises an exception if not configured
Refer to flask_oidc docs for OpenIDConnect configuration (make sure that the installed version matches the version described in the docs).
You can set the accept_token decorator for all methods:
build_resource(User, user_ns, get_model, upsert_model, oidc=app.oidc)
Or specify the accept_token decorator for each method:
build_resource(User, user_ns, get_model, upsert_model, get_all_oidc=app.oidc)
Available arguments:
get_all_oidcget_by_id_oidcpost_oidcput_oidcdelete_oidc
To introduce other decorators please create your own resource and set the required decorators.
Refer to flask_restx docs for decorators configuration (make sure that the installed version matches the version described in the docs).
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file eaasy-0.4.3.tar.gz.
File metadata
- Download URL: eaasy-0.4.3.tar.gz
- Upload date:
- Size: 18.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ba50b7087e064d645d9ddd8386d24d7a72905289c277d75b6db809a40baddea6
|
|
| MD5 |
23cc89617a89209e3673aae062a78d69
|
|
| BLAKE2b-256 |
c5daa796b174c1623ef3cb3cf4ebb256a8b4f42e55618961f13cb137cc93bbb5
|
Provenance
The following attestation bundles were made for eaasy-0.4.3.tar.gz:
Publisher:
CD.yml on ciuliene/eaasy
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
eaasy-0.4.3.tar.gz -
Subject digest:
ba50b7087e064d645d9ddd8386d24d7a72905289c277d75b6db809a40baddea6 - Sigstore transparency entry: 928214814
- Sigstore integration time:
-
Permalink:
ciuliene/eaasy@d810d19f2343d63c7638a80806c5888d1e06b00c -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ciuliene
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
CD.yml@d810d19f2343d63c7638a80806c5888d1e06b00c -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file eaasy-0.4.3-py3-none-any.whl.
File metadata
- Download URL: eaasy-0.4.3-py3-none-any.whl
- Upload date:
- Size: 16.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f219f58546a6b1c33b73c0417ce5b862964b6981708495f32e1e47dd20f6ca37
|
|
| MD5 |
e1199a05e242ecfb7bf4e4af8baaa381
|
|
| BLAKE2b-256 |
c57adb46b8c1a91498f7b4b72e99ff79993cd5cc112f63c8118b1334e50a5878
|
Provenance
The following attestation bundles were made for eaasy-0.4.3-py3-none-any.whl:
Publisher:
CD.yml on ciuliene/eaasy
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
eaasy-0.4.3-py3-none-any.whl -
Subject digest:
f219f58546a6b1c33b73c0417ce5b862964b6981708495f32e1e47dd20f6ca37 - Sigstore transparency entry: 928214815
- Sigstore integration time:
-
Permalink:
ciuliene/eaasy@d810d19f2343d63c7638a80806c5888d1e06b00c -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ciuliene
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
CD.yml@d810d19f2343d63c7638a80806c5888d1e06b00c -
Trigger Event:
workflow_dispatch
-
Statement type: