/django-simplewallet

A simple wallet app for Django, with a test portal to show

Primary LanguagePythonGNU General Public License v3.0GPL-3.0

django-simplewallet

A simple wallet django app.

Compatibility

This project is compatible with Python 3 and Django 3 (tested up to Django 3.2).

Structure

The project is divided in two apps. The most important is wallet that manages wallets.

The app toystore implements a simple shop where the user can buy objects; its code can be used and adapted to your needs.

Running

Install django. Then, at least once, run

$ cd src
$ python3 ./manage.py migrate
$ python3 ./helper.py create_fake_users
# python3 ./manage.py deposit --username=foobar --amount=100

then

$ python3 ./manage.py runserver

this will use a sqlite database for convenience.

Connect to localhost:8000 , authenticate with username foobar and password barfoo, and buy an object.

Running with docker

The project contains a portal called toystore that demoes the wallet app.

Install docker and docker-compose. Use your distribution package manager, or

$ curl -sSL https://get.docker.com
$ pip install docker-compose  # run it as sudo to install it globally.
$ gpasswd -a $USER docker  # add user to docker group
# login/logout from shell

Run the project!

$ docker-compose up

Connect to localhost:8000 , authenticate with username foobar and password barfoo, and buy an object.

Testing

The project contains tests, but some fail with sqlite; whereas they work fine with mysql.

To properly test the code, just start the docker container: all tests are run prior to starting the server.

Installing the app

There is no provision to install this app: just symlink the directory src/wallet into the root of your Django project; then add it to the INSTALLED_APPS in settings.py

INSTALLED_APPS += [ 'wallet', ]

and add its urls into the main urls.py

urlpatterns += [ path('wallet/', include('wallet.urls')),]

Creating a New Wallet

A wallet is owned by a user. Should you be using a custom user model, the wallet should still work properly as the wallet points to settings.AUTH_USER_MODEL.

from wallet.models import Wallet

# wallets are owned by users.
wallet = user.wallet_set.create()

Operate

The wallet model has three methods deposit, withdraw and transfer. These are low-level methods; in particular, they should be wrapped inside a transaction; see below for examples.

There are also higher level functions by the same name in wallet.utils : there the user can be specified as User class, by username or by email; groups of users can be specified; and operations are protected by transactions.

These functions can also be accessed from command line, using the manage.py script.

Permissions

To withdraw/deposit on a wallet, the user must have the operate permission.

To transfer between wallets, both the sending and receiving user must have operate permission.

Depositing a balance to a wallet

from django.db import transaction

with transaction.atomic():
    # We need to lock the wallet first so that we're sure
    # that nobody modifies the wallet at the same time 
    # we're modifying it.
    wallet = Wallet.select_for_update().get(pk=wallet.id)
    wallet.deposit(100)  # amount

Withdrawing a balance from a wallet

from django.db import transaction

with transaction.atomic():
    # We need to lock the wallet first so that we're sure
    # that nobody modifies the wallet at the same time 
    # we're modifying it.
    wallet = Wallet.select_for_update().get(pk=wallet.id)
    wallet.withdraw(100)  # amount

Withdrawing with an insufficient balance

When a user tries to withdraw from a wallet with an amount greater than its balance, the transaction raises a wallet.errors.InsufficientBalance error.

# wallet.current_balance  # 50

# This raises an wallet.errors.InsufficentBalance.
wallet.withdraw(100)

This error inherits from django.db.IntegrityError so that when it is raised, the whole transaction is automatically rolled-back.

Transferring between wallets.

One can transfer a values between wallets. It uses withdraw and deposit internally. Should the sending wallet have an insufficient balance, wallet.errors.InsufficientBalance is raised.

with transaction.atomic():
    wallet = Wallet.select_for_update().get(pk=wallet_id)
    transfer_to_wallet = Wallet.select_for_update().get(pk=transfer_to_wallet_id)
    wallet.transfer(transfer_to_wallet, 100)

Buying contracts

The app can prepare a contract for the user, that can then accept the terms (and pay) or refuse.

This contract contains all relevant informations; in particular, it contains an instance of a function that, upon payment, performs all needed operations to finalize.

The contract is encoded, and passed to user's browser, either in the URL, or in a form; the first method though may fail sometimes (e.g. some versions of Apache and WSGI silently fail on long URLs).

The workflow is simple, look into toystore/views.py.

CURRENCY_STORE_FIELD

The CURRENCY_STORE_FIELD is a django field class that contains how the fields should be stored. By default, it uses django.models.BigIntegerField. It was chosen that way for simplicity - just make cents into your smallest unit (0.01 -> 1, 1.00 -> 100).

You can change this to decimal by adding this to your settings.py:

# settings.py
CURRENCY_STORE_FIELD = models.DecimalField(max_digits=10, decimal_places=2)

You need to run ./manage.py makemigrations after that.

History

This project is a fork of https://github.com/thejpanganiban/django-pursed by Jesse Panganiban