Compare commits

..

1 Commits

Author SHA1 Message Date
Raphael Michel
d169958687 Add Transaction.source flag 2022-10-19 12:09:55 +02:00
433 changed files with 215653 additions and 322355 deletions

View File

@@ -9,7 +9,6 @@ updates:
directory: "/src" directory: "/src"
schedule: schedule:
interval: "daily" interval: "daily"
versioning-strategy: increase
- package-ecosystem: "npm" - package-ecosystem: "npm"
directory: "/src/pretix/static/npm_dir" directory: "/src/pretix/static/npm_dir"
schedule: schedule:

View File

@@ -14,22 +14,16 @@ on:
- 'src/pretix/static/**' - 'src/pretix/static/**'
- 'src/tests/**' - 'src/tests/**'
permissions:
contents: read # to fetch code (actions/checkout)
env:
FORCE_COLOR: 1
jobs: jobs:
spelling: spelling:
name: Spellcheck name: Spellcheck
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- uses: actions/cache@v1 - uses: actions/cache@v1
with: with:
path: ~/.cache/pip path: ~/.cache/pip
@@ -37,7 +31,7 @@ jobs:
restore-keys: | restore-keys: |
${{ runner.os }}-pip- ${{ runner.os }}-pip-
- name: Install system packages - name: Install system packages
run: sudo apt update && sudo apt install enchant-2 hunspell aspell-en run: sudo apt update && sudo apt install enchant hunspell aspell-en
- name: Install Dependencies - name: Install Dependencies
run: pip3 install -Ur requirements.txt run: pip3 install -Ur requirements.txt
working-directory: ./doc working-directory: ./doc

View File

@@ -12,22 +12,16 @@ on:
- 'doc/**' - 'doc/**'
- 'src/pretix/locale/**' - 'src/pretix/locale/**'
permissions:
contents: read # to fetch code (actions/checkout)
env:
FORCE_COLOR: 1
jobs: jobs:
compile: compile:
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
name: Check gettext syntax name: Check gettext syntax
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- uses: actions/cache@v1 - uses: actions/cache@v1
with: with:
path: ~/.cache/pip path: ~/.cache/pip
@@ -46,14 +40,14 @@ jobs:
run: python manage.py compilejsi18n run: python manage.py compilejsi18n
working-directory: ./src working-directory: ./src
spelling: spelling:
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
name: Spellcheck name: Spellcheck
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- uses: actions/cache@v1 - uses: actions/cache@v1
with: with:
path: ~/.cache/pip path: ~/.cache/pip
@@ -61,7 +55,7 @@ jobs:
restore-keys: | restore-keys: |
${{ runner.os }}-pip- ${{ runner.os }}-pip-
- name: Install system packages - name: Install system packages
run: sudo apt update && sudo apt install enchant-2 hunspell hunspell-de-de aspell-en aspell-de run: sudo apt update && sudo apt install enchant hunspell hunspell-de-de aspell-en aspell-de
- name: Install Dependencies - name: Install Dependencies
run: pip3 install -e ".[dev]" run: pip3 install -e ".[dev]"
working-directory: ./src working-directory: ./src

View File

@@ -12,22 +12,16 @@ on:
- 'src/pretix/locale/**' - 'src/pretix/locale/**'
- 'src/pretix/static/**' - 'src/pretix/static/**'
permissions:
contents: read # to fetch code (actions/checkout)
env:
FORCE_COLOR: 1
jobs: jobs:
isort: isort:
name: isort name: isort
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- uses: actions/cache@v1 - uses: actions/cache@v1
with: with:
path: ~/.cache/pip path: ~/.cache/pip
@@ -42,13 +36,13 @@ jobs:
working-directory: ./src working-directory: ./src
flake: flake:
name: flake8 name: flake8
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- uses: actions/cache@v1 - uses: actions/cache@v1
with: with:
path: ~/.cache/pip path: ~/.cache/pip
@@ -63,13 +57,13 @@ jobs:
working-directory: ./src working-directory: ./src
licenseheader: licenseheader:
name: licenseheaders name: licenseheaders
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: Set up Python 3.9 - name: Set up Python 3.8
uses: actions/setup-python@v1 uses: actions/setup-python@v1
with: with:
python-version: 3.9 python-version: 3.8
- name: Install Dependencies - name: Install Dependencies
run: pip3 install licenseheaders run: pip3 install licenseheaders
- name: Run licenseheaders - name: Run licenseheaders

View File

@@ -12,29 +12,23 @@ on:
- 'doc/**' - 'doc/**'
- 'src/pretix/locale/**' - 'src/pretix/locale/**'
permissions:
contents: read # to fetch code (actions/checkout)
env:
FORCE_COLOR: 1
jobs: jobs:
test: test:
runs-on: ubuntu-22.04 runs-on: ubuntu-latest
name: Tests name: Tests
strategy: strategy:
matrix: matrix:
python-version: ["3.7", "3.9", "3.10"] python-version: ["3.7", "3.8", "3.9"]
database: [sqlite, postgres, mysql] database: [sqlite, postgres, mysql]
exclude: exclude:
- database: mysql - database: mysql
python-version: "3.10" python-version: "3.8"
- database: mysql - database: mysql
python-version: "3.9" python-version: "3.9"
- database: sqlite - database: sqlite
python-version: "3.7" python-version: "3.7"
- database: sqlite - database: sqlite
python-version: "3.10" python-version: "3.8"
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- uses: getong/mariadb-action@v1.1 - uses: getong/mariadb-action@v1.1
@@ -61,9 +55,9 @@ jobs:
restore-keys: | restore-keys: |
${{ runner.os }}-pip- ${{ runner.os }}-pip-
- name: Install system dependencies - name: Install system dependencies
run: sudo apt update && sudo apt install gettext mariadb-client run: sudo apt update && sudo apt install gettext mariadb-client-10.3
- name: Install Python dependencies - name: Install Python dependencies
run: pip3 install --ignore-requires-python -e ".[dev]" mysqlclient psycopg2-binary # We ignore that flake8 needs newer python as we don't run flake8 during tests run: pip3 install -e ".[dev]" mysqlclient psycopg2-binary
working-directory: ./src working-directory: ./src
- name: Run checks - name: Run checks
run: python manage.py check run: python manage.py check
@@ -81,6 +75,5 @@ jobs:
uses: codecov/codecov-action@v1 uses: codecov/codecov-action@v1
with: with:
file: src/coverage.xml file: src/coverage.xml
token: ${{ secrets.CODECOV_TOKEN }}
fail_ci_if_error: true fail_ci_if_error: true
if: matrix.database == 'postgres' && matrix.python-version == '3.10' if: matrix.database == 'postgres' && matrix.python-version == '3.8'

View File

@@ -117,9 +117,6 @@ Example::
``loglevel`` ``loglevel``
Set console and file log level (``DEBUG``, ``INFO``, ``WARNING``, ``ERROR`` or ``CRITICAL``). Defaults to ``INFO``. Set console and file log level (``DEBUG``, ``INFO``, ``WARNING``, ``ERROR`` or ``CRITICAL``). Defaults to ``INFO``.
``request_id_header``
Specifies the name of a header that should be used for logging request IDs. Off by default.
Locale settings Locale settings
--------------- ---------------
@@ -141,7 +138,7 @@ Database settings
Example:: Example::
[database] [database]
backend=postgresql backend=mysql
name=pretix name=pretix
user=pretix user=pretix
password=abcd password=abcd
@@ -149,7 +146,7 @@ Example::
port=3306 port=3306
``backend`` ``backend``
One of ``mysql`` (deprecated), ``sqlite3`` and ``postgresql``. One of ``mysql``, ``sqlite3``, ``oracle`` and ``postgresql``.
Default: ``sqlite3``. Default: ``sqlite3``.
If you use MySQL, be sure to create your database using If you use MySQL, be sure to create your database using
@@ -163,7 +160,7 @@ Example::
Connection details for the database connection. Empty by default. Connection details for the database connection. Empty by default.
``galera`` ``galera``
(Deprecated) Indicates if the database backend is a MySQL/MariaDB Galera cluster and Indicates if the database backend is a MySQL/MariaDB Galera cluster and
turns on some optimizations/special case handlers. Default: ``False`` turns on some optimizations/special case handlers. Default: ``False``
.. _`config-replica`: .. _`config-replica`:
@@ -194,7 +191,7 @@ Example::
[urls] [urls]
media=/media/ media=/media/
static=/static/ static=/media/
``media`` ``media``
The URL to be used to serve user-uploaded content. You should not need to modify The URL to be used to serve user-uploaded content. You should not need to modify
@@ -399,9 +396,9 @@ The two ``transport_options`` entries can be omitted in most cases.
If they are present they need to be a valid JSON dictionary. If they are present they need to be a valid JSON dictionary.
For possible entries in that dictionary see the `Celery documentation`_. For possible entries in that dictionary see the `Celery documentation`_.
To use redis with sentinels set the broker or backend to ``sentinel://sentinel_host_1:26379;sentinel_host_2:26379/0`` To use redis with sentinels set the broker or backend to ``sentinel://sentinel_host_1:26379;sentinal_host_2:26379/0``
and the respective transport_options to ``{"master_name":"mymaster"}``. and the respective transport_options to ``{"master_name":"mymaster"}``.
If your redis instances behind the sentinel have a password use ``sentinel://:my_password@sentinel_host_1:26379;sentinel_host_2:26379/0``. If your redis instances behind the sentinel have a password use ``sentinel://:my_password@sentinel_host_1:26379;sentinal_host_2:26379/0``.
If your redis sentinels themselves have a password set the transport_options to ``{"master_name":"mymaster","sentinel_kwargs":{"password":"my_password"}}``. If your redis sentinels themselves have a password set the transport_options to ``{"master_name":"mymaster","sentinel_kwargs":{"password":"my_password"}}``.
Sentry Sentry

View File

@@ -14,5 +14,4 @@ This documentation is for everyone who wants to install pretix on a server.
maintainance maintainance
scaling scaling
errors errors
mysql2postgres
indexes indexes

View File

@@ -14,7 +14,7 @@ This has some trade-offs in terms of performance and isolation but allows a rath
get it right. If you're not feeling comfortable managing a Linux server, check out our hosting and service get it right. If you're not feeling comfortable managing a Linux server, check out our hosting and service
offers at `pretix.eu`_. offers at `pretix.eu`_.
We tested this guide on the Linux distribution **Debian 11.0** but it should work very similar on other We tested this guide on the Linux distribution **Debian 8.0** but it should work very similar on other
modern distributions, especially on all systemd-based ones. modern distributions, especially on all systemd-based ones.
Requirements Requirements
@@ -26,7 +26,7 @@ installation guides):
* `Docker`_ * `Docker`_
* A SMTP server to send out mails, e.g. `Postfix`_ on your machine or some third-party server you have credentials for * A SMTP server to send out mails, e.g. `Postfix`_ on your machine or some third-party server you have credentials for
* A HTTP reverse proxy, e.g. `nginx`_ or Apache to allow HTTPS connections * A HTTP reverse proxy, e.g. `nginx`_ or Apache to allow HTTPS connections
* A `PostgreSQL`_ 9.6+ database server * A `PostgreSQL`_ 9.6+, `MySQL`_ 5.7+, or MariaDB 10.2.7+ database server
* A `redis`_ server * A `redis`_ server
We also recommend that you use a firewall, although this is not a pretix-specific recommendation. If you're new to We also recommend that you use a firewall, although this is not a pretix-specific recommendation. If you're new to
@@ -58,6 +58,9 @@ directory writable to the user that runs pretix inside the docker container::
Database Database
-------- --------
.. warning:: **Please use PostgreSQL for all new installations**. If you need to go for MySQL, make sure you run
**MySQL 5.7 or newer** or **MariaDB 10.2.7 or newer**.
Next, we need a database and a database user. We can create these with any kind of database managing tool or directly on Next, we need a database and a database user. We can create these with any kind of database managing tool or directly on
our database's shell. Please make sure that UTF8 is used as encoding for the best compatibility. You can check this with our database's shell. Please make sure that UTF8 is used as encoding for the best compatibility. You can check this with
the following command:: the following command::
@@ -83,6 +86,13 @@ Restart PostgreSQL after you changed these files::
If you have a firewall running, you should also make sure that port 5432 is reachable from the ``172.17.0.1/16`` subnet. If you have a firewall running, you should also make sure that port 5432 is reachable from the ``172.17.0.1/16`` subnet.
For MySQL, you can either also use network-based connections or mount the ``/var/run/mysqld/mysqld.sock`` socket into the docker container.
When using MySQL, make sure you set the character set of the database to ``utf8mb4``, e.g. like this::
mysql > CREATE DATABASE pretix DEFAULT CHARACTER SET utf8mb4 DEFAULT COLLATE utf8mb4_unicode_ci;
You will also need to make sure that ``sql_mode`` in your ``my.cnf`` file does **not** include ``ONLY_FULL_GROUP_BY``.
Redis Redis
----- -----
@@ -142,13 +152,15 @@ Fill the configuration file ``/etc/pretix/pretix.cfg`` with the following conten
trust_x_forwarded_proto=on trust_x_forwarded_proto=on
[database] [database]
; Replace postgresql with mysql for MySQL
backend=postgresql backend=postgresql
name=pretix name=pretix
user=pretix user=pretix
; Replace with the password you chose above ; Replace with the password you chose above
password=********* password=*********
; In most docker setups, 172.17.0.1 is the address of the docker host. Adjust ; In most docker setups, 172.17.0.1 is the address of the docker host. Adjust
; this to wherever your database is running, e.g. the name of a linked container. ; this to wherever your database is running, e.g. the name of a linked container
; or of a mounted MySQL socket.
host=172.17.0.1 host=172.17.0.1
[mail] [mail]
@@ -200,6 +212,8 @@ named ``/etc/systemd/system/pretix.service`` with the following content::
[Install] [Install]
WantedBy=multi-user.target WantedBy=multi-user.target
When using MySQL and socket mounting, you'll need the additional flag ``-v /var/run/mysqld:/var/run/mysqld`` in the command.
You can now run the following commands You can now run the following commands
to enable and start the service:: to enable and start the service::
@@ -325,6 +339,7 @@ workers, e.g. ``docker run … taskworker -Q notifications --concurrency 32``.
.. _nginx: https://botleg.com/stories/https-with-lets-encrypt-and-nginx/ .. _nginx: https://botleg.com/stories/https-with-lets-encrypt-and-nginx/
.. _Let's Encrypt: https://letsencrypt.org/ .. _Let's Encrypt: https://letsencrypt.org/
.. _pretix.eu: https://pretix.eu/ .. _pretix.eu: https://pretix.eu/
.. _MySQL: https://dev.mysql.com/doc/refman/5.7/en/linux-installation-apt-repo.html
.. _PostgreSQL: https://www.digitalocean.com/community/tutorials/how-to-install-and-use-postgresql-on-ubuntu-20-04 .. _PostgreSQL: https://www.digitalocean.com/community/tutorials/how-to-install-and-use-postgresql-on-ubuntu-20-04
.. _redis: https://blog.programster.org/debian-8-install-redis-server/ .. _redis: https://blog.programster.org/debian-8-install-redis-server/
.. _ufw: https://en.wikipedia.org/wiki/Uncomplicated_Firewall .. _ufw: https://en.wikipedia.org/wiki/Uncomplicated_Firewall

View File

@@ -12,7 +12,7 @@ solution with many things readily set-up, look at :ref:`dockersmallscale`.
get it right. If you're not feeling comfortable managing a Linux server, check out our hosting and service get it right. If you're not feeling comfortable managing a Linux server, check out our hosting and service
offers at `pretix.eu`_. offers at `pretix.eu`_.
We tested this guide on the Linux distribution **Debian 11.6** but it should work very similar on other We tested this guide on the Linux distribution **Debian 10.0** but it should work very similar on other
modern distributions, especially on all systemd-based ones. modern distributions, especially on all systemd-based ones.
Requirements Requirements
@@ -23,7 +23,7 @@ installation guides):
* A SMTP server to send out mails, e.g. `Postfix`_ on your machine or some third-party server you have credentials for * A SMTP server to send out mails, e.g. `Postfix`_ on your machine or some third-party server you have credentials for
* A HTTP reverse proxy, e.g. `nginx`_ or Apache to allow HTTPS connections * A HTTP reverse proxy, e.g. `nginx`_ or Apache to allow HTTPS connections
* A `PostgreSQL`_ 9.6+ database server * A `PostgreSQL`_ 9.6+, `MySQL`_ 5.7+, or MariaDB 10.2.7+ database server
* A `redis`_ server * A `redis`_ server
* A `nodejs`_ installation * A `nodejs`_ installation
@@ -47,6 +47,9 @@ In this guide, all code lines prepended with a ``#`` symbol are commands that yo
Database Database
-------- --------
.. warning:: **Please use PostgreSQL for all new installations**. If you need to go for MySQL, make sure you run
**MySQL 5.7 or newer** or **MariaDB 10.2.7 or newer**.
Having the database server installed, we still need a database and a database user. We can create these with any kind Having the database server installed, we still need a database and a database user. We can create these with any kind
of database managing tool or directly on our database's shell. Please make sure that UTF8 is used as encoding for the of database managing tool or directly on our database's shell. Please make sure that UTF8 is used as encoding for the
best compatibility. You can check this with the following command:: best compatibility. You can check this with the following command::
@@ -58,6 +61,12 @@ For PostgreSQL database creation, we would do::
# sudo -u postgres createuser pretix # sudo -u postgres createuser pretix
# sudo -u postgres createdb -O pretix pretix # sudo -u postgres createdb -O pretix pretix
When using MySQL, make sure you set the character set of the database to ``utf8mb4``, e.g. like this::
mysql > CREATE DATABASE pretix DEFAULT CHARACTER SET utf8mb4 DEFAULT COLLATE utf8mb4_unicode_ci;
You will also need to make sure that ``sql_mode`` in your ``my.cnf`` file does **not** include ``ONLY_FULL_GROUP_BY``.
Package dependencies Package dependencies
-------------------- --------------------
@@ -65,7 +74,7 @@ To build and run pretix, you will need the following debian packages::
# apt-get install git build-essential python-dev python3-venv python3 python3-pip \ # apt-get install git build-essential python-dev python3-venv python3 python3-pip \
python3-dev libxml2-dev libxslt1-dev libffi-dev zlib1g-dev libssl-dev \ python3-dev libxml2-dev libxslt1-dev libffi-dev zlib1g-dev libssl-dev \
gettext libpq-dev libjpeg-dev libopenjp2-7-dev gettext libpq-dev libmariadb-dev libjpeg-dev libopenjp2-7-dev
Config file Config file
----------- -----------
@@ -88,12 +97,16 @@ Fill the configuration file ``/etc/pretix/pretix.cfg`` with the following conten
trust_x_forwarded_proto=on trust_x_forwarded_proto=on
[database] [database]
; For MySQL, replace with "mysql"
backend=postgresql backend=postgresql
name=pretix name=pretix
user=pretix user=pretix
; For PostgreSQL on the same host, we don't need a password because we can use ; For MySQL, enter the user password. For PostgreSQL on the same host,
; peer authentication if our PostgreSQL user matches our unix user. ; we don't need one because we can use peer authentification if our
; PostgreSQL user matches our unix user.
password= password=
; For MySQL, use local socket, e.g. /var/run/mysqld/mysqld.sock
; For a remote host, supply an IP address
; For local postgres authentication, you can leave it empty ; For local postgres authentication, you can leave it empty
host= host=
@@ -127,6 +140,10 @@ We now install pretix, its direct dependencies and gunicorn::
(venv)$ pip3 install pretix gunicorn (venv)$ pip3 install pretix gunicorn
If you're running MySQL, also install the client library::
(venv)$ pip3 install mysqlclient
Note that you need Python 3.7 or newer. You can find out your Python version using ``python -V``. Note that you need Python 3.7 or newer. You can find out your Python version using ``python -V``.
We also need to create a data directory:: We also need to create a data directory::
@@ -301,32 +318,12 @@ example::
(venv)$ python -m pretix rebuild (venv)$ python -m pretix rebuild
# systemctl restart pretix-web pretix-worker # systemctl restart pretix-web pretix-worker
System updates
--------------
After system updates, such as updates to a new Ubuntu or Debian release, you might be using a new Python version.
That's great, but requires some adjustments. First, adjust any old version paths in your nginx configuration file.
Then, re-create your Python environment::
$ source /var/pretix/venv/bin/activate
(venv)$ pip3 freeze > /tmp/pip-backup.txt
$ rm -rf /var/pretix/venv
$ python3 -m venv /var/pretix/venv
$ source /var/pretix/venv/bin/activate
(venv)$ pip3 install -U pip wheel setuptools
(venv)$ pip3 install -r /tmp/pip-backup.txt
Then, proceed like after any plugin installation::
(venv)$ python -m pretix migrate
(venv)$ python -m pretix rebuild
(venv)$ python -m pretix updatestyles
# systemctl restart pretix-web pretix-worker
.. _Postfix: https://www.digitalocean.com/community/tutorials/how-to-install-and-configure-postfix-as-a-send-only-smtp-server-on-ubuntu-16-04 .. _Postfix: https://www.digitalocean.com/community/tutorials/how-to-install-and-configure-postfix-as-a-send-only-smtp-server-on-ubuntu-16-04
.. _nginx: https://botleg.com/stories/https-with-lets-encrypt-and-nginx/ .. _nginx: https://botleg.com/stories/https-with-lets-encrypt-and-nginx/
.. _Let's Encrypt: https://letsencrypt.org/ .. _Let's Encrypt: https://letsencrypt.org/
.. _pretix.eu: https://pretix.eu/ .. _pretix.eu: https://pretix.eu/
.. _MySQL: https://dev.mysql.com/doc/refman/5.7/en/linux-installation-apt-repo.html
.. _PostgreSQL: https://www.digitalocean.com/community/tutorials/how-to-install-and-use-postgresql-on-ubuntu-20-04 .. _PostgreSQL: https://www.digitalocean.com/community/tutorials/how-to-install-and-use-postgresql-on-ubuntu-20-04
.. _redis: https://blog.programster.org/debian-8-install-redis-server/ .. _redis: https://blog.programster.org/debian-8-install-redis-server/
.. _ufw: https://en.wikipedia.org/wiki/Uncomplicated_Firewall .. _ufw: https://en.wikipedia.org/wiki/Uncomplicated_Firewall

View File

@@ -17,11 +17,11 @@ Backups
There are essentially two things which you should create backups of: There are essentially two things which you should create backups of:
Database Database
Your SQL database. This is critical and you should **absolutely always create automatic Your SQL database (MySQL or PostgreSQL). This is critical and you should **absolutely
backups of your database**. There are tons of tutorials on the internet on how to do this, always create automatic backups of your database**. There are tons of tutorials on the
and the exact process depends on the choice of your database. For PostgreSQL, see the internet on how to do this, and the exact process depends on the choice of your database.
``pg_dump`` tool. You probably want to create a cronjob that does the backups for you on a For MySQL, see ``mysqldump`` and for PostgreSQL, see the ``pg_dump`` tool. You probably
regular schedule. want to create a cronjob that does the backups for you on a regular schedule.
Data directory Data directory
The data directory of your pretix configuration might contain some things that you should The data directory of your pretix configuration might contain some things that you should

View File

@@ -1,148 +0,0 @@
.. highlight:: none
Migrating from MySQL/MariaDB to PostgreSQL
==========================================
Our recommended database for all production installations is PostgreSQL. Support for MySQL/MariaDB will be removed in
pretix 5.0.
In order to follow this guide, your pretix installation needs to be a version that fully supports MySQL/MariaDB. If you
already upgraded to pretix 5.0, downgrade back to the last 4.x release using ``pip``.
.. note:: We have tested this guide carefully, but we can't assume any liability for its correctness. The data loss
risk should be low as long as pretix is not running while you do the migration. If you are a pretix Enterprise
customer, feel free to reach out in advance if you want us to support you along the way.
Update database schema
----------------------
Before you start, make sure your database schema is up to date::
# sudo -u pretix -s
$ source /var/pretix/venv/bin/activate
(venv)$ python -m pretix migrate
Install PostgreSQL
------------------
Now, install and set up a PostgreSQL server. For a local installation on Debian or Ubuntu, use::
# apt install postgresql
Having the database server installed, we still need a database and a database user. We can create these with any kind
of database managing tool or directly on our database's shell. Please make sure that UTF8 is used as encoding for the
best compatibility. You can check this with the following command::
# sudo -u postgres psql -c 'SHOW SERVER_ENCODING'
Without Docker
""""""""""""""
For our standard manual installation, create the database and user like this::
# sudo -u postgres createuser pretix
# sudo -u postgres createdb -O pretix pretix
With Docker
"""""""""""
For our standard docker installation, create the database and user like this::
# sudo -u postgres createuser -P pretix
# sudo -u postgres createdb -O pretix pretix
Make sure that your database listens on the network. If PostgreSQL on the same same host as docker, but not inside a docker container, we recommend that you just listen on the Docker interface by changing the following line in ``/etc/postgresql/<version>/main/postgresql.conf``::
listen_addresses = 'localhost,172.17.0.1'
You also need to add a new line to ``/etc/postgresql/<version>/main/pg_hba.conf`` to allow network connections to this user and database::
host pretix pretix 172.17.0.1/16 md5
Restart PostgreSQL after you changed these files::
# systemctl restart postgresql
If you have a firewall running, you should also make sure that port 5432 is reachable from the ``172.17.0.1/16`` subnet.
Of course, instead of all this you can also run a PostgreSQL docker container and link it to the pretix container.
Stop pretix
-----------
To prevent any more changes to your data, stop pretix from running::
# systemctl stop pretix-web pretix-worker
Change configuration
--------------------
Change the database configuration in your ``/etc/pretix/pretix.cfg`` file::
[database]
backend=postgresql
name=pretix
user=pretix
password= ; only required for docker or remote database, can be kept empty for local auth
host= ; set to 172.17.0.1 in docker setup, keep empty for local auth
Create database schema
-----------------------
To create the schema in your new PostgreSQL database, use the following commands::
# sudo -u pretix -s
$ source /var/pretix/venv/bin/activate
(venv)$ python -m pretix migrate
Migrate your data
-----------------
Install ``pgloader``::
# apt install pgloader
Create a new file ``/tmp/pretix.load``, replacing the MySQL and PostgreSQL connection strings with the correct user names, passwords, and/or database names::
LOAD DATABASE
FROM mysql://pretix:password@localhost/pretix -- replace with mysql://username:password@hostname/dbname
INTO postgresql:///pretix -- replace with dbname
WITH data only, include no drop, truncate, disable triggers,
create no indexes, drop indexes, reset sequences
ALTER SCHEMA 'pretix' RENAME TO 'public' -- replace pretix with the name of the MySQL database
ALTER TABLE NAMES MATCHING ~/.*/
SET SCHEMA 'public'
SET timezone TO '+00:00'
SET PostgreSQL PARAMETERS
maintenance_work_mem to '128MB',
work_mem to '12MB';
Then, run::
# sudo -u postgres pgloader /tmp/pretix.load
The output should end with a table summarizing the results for every table. You can ignore warnings about type casts
and missing constraints.
Afterwards, delete the file again::
# rm -rf /tmp/pretix.load
Start pretix
------------
Now, restart pretix. Maybe stop your MySQL server as a verification step that you are no longer using it::
# systemctl stop mariadb
# systemctl start pretix-web pretix-worker
And you're done! After you've verified everything has been copied correctly, you can delete the old MySQL database.
.. note:: Don't forget to update your backup process to back up your PostgreSQL database instead of your MySQL database now.

View File

@@ -42,7 +42,7 @@ A pretix installation usually consists of the following components which run per
* ``pretix-worker`` is a Celery-based application that processes tasks that should be run asynchronously outside of the web application process. * ``pretix-worker`` is a Celery-based application that processes tasks that should be run asynchronously outside of the web application process.
* A **PostgreSQL database** keeps all the important data and processes the actual transactions. * A **SQL database** keeps all the important data and processes the actual transactions. We recommend using PostgreSQL, but MySQL/MariaDB works as well.
* A **web server** that terminates TLS and HTTP connections and forwards them to ``pretix-web``. In some cases, e.g. when serving static files, the web servers might return a response directly. We recommend using ``nginx``. * A **web server** that terminates TLS and HTTP connections and forwards them to ``pretix-web``. In some cases, e.g. when serving static files, the web servers might return a response directly. We recommend using ``nginx``.
@@ -74,7 +74,7 @@ We recommend reading up on tuning your web server for high concurrency. For ngin
processes and the number of connections each worker process accepts. Double-check that TLS session caching works, because TLS processes and the number of connections each worker process accepts. Double-check that TLS session caching works, because TLS
handshakes can get really expensive. handshakes can get really expensive.
During a traffic peak, your web server will be able to make use of more CPU resources, while memory usage will stay comparatively low, During a traffic peak, your web server will be able to make us of more CPU resources, while memory usage will stay comparatively low,
so if you invest in more hardware here, invest in more and faster CPU cores. so if you invest in more hardware here, invest in more and faster CPU cores.
Make sure that pretix' static files (such as CSS and JavaScript assets) as well as user-uploaded media files (event logos, etc) Make sure that pretix' static files (such as CSS and JavaScript assets) as well as user-uploaded media files (event logos, etc)

View File

@@ -48,11 +48,10 @@ Possible permissions are:
Compatibility Compatibility
------------- -------------
We try to avoid any breaking changes to our API to avoid hassle on your end. If possible, we'll We currently see pretix' API as a beta-stage feature. We therefore do not give any guarantees
build new features in a way that keeps all pre-existing API usage unchanged. In some cases, for compatibility between feature releases of pretix (such as 1.5 and 1.6). However, as always,
this might not be possible or only possible with restrictions. In these case, any we try not to break things when we don't need to. Any backwards-incompatible changes will be
backwards-incompatible changes will be prominently noted in the "Changes to the REST API" prominently noted in the release notes.
section of our release notes. If possible, we will announce them multiple releases in advance.
We treat the following types of changes as *backwards-compatible* so we ask you to make sure We treat the following types of changes as *backwards-compatible* so we ask you to make sure
that your clients can deal with them properly: that your clients can deal with them properly:
@@ -61,7 +60,6 @@ that your clients can deal with them properly:
* Support of new HTTP methods for a given API endpoint * Support of new HTTP methods for a given API endpoint
* Support of new query parameters for a given API endpoint * Support of new query parameters for a given API endpoint
* New fields contained in API responses * New fields contained in API responses
* New possible values of enumeration-like fields
* Response body structure or message texts on failed requests (``4xx``, ``5xx`` response codes) * Response body structure or message texts on failed requests (``4xx``, ``5xx`` response codes)
We treat the following types of changes as *backwards-incompatible*: We treat the following types of changes as *backwards-incompatible*:
@@ -192,9 +190,6 @@ Relative date *either* String in ISO 8601 ``"2017-12-27"``,
File URL in responses, ``file:`` ``"https://…"``, ``"file:…"`` File URL in responses, ``file:`` ``"https://…"``, ``"file:…"``
specifiers in requests specifiers in requests
(see below). (see below).
Date range *either* two dates separated ``2022-03-18/2022-03-23``, ``2022-03-18/``,
by ``/`` *or* the name of a ``/2022-03-23``, ``week_this``, ``week_next``,
defined range. ``month_this``
===================== ============================ =================================== ===================== ============================ ===================================
Query parameters Query parameters

View File

@@ -43,6 +43,10 @@ seat objects The assigned se
└ seat_guid string Identifier of the seat within the seating plan └ seat_guid string Identifier of the seat within the seating plan
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.0
This ``seat`` attribute has been added.
.. versionchanged:: 4.14 .. versionchanged:: 4.14
This ``is_bundled`` attribute has been added and the cart creation endpoints have been updated. This ``is_bundled`` attribute has been added and the cart creation endpoints have been updated.

View File

@@ -39,6 +39,23 @@ exit_all_at datetime Automatically c
addon_match boolean If ``true``, tickets on this list can be redeemed by scanning their parent ticket if this still leads to an unambiguous match. addon_match boolean If ``true``, tickets on this list can be redeemed by scanning their parent ticket if this still leads to an unambiguous match.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.9
The ``subevent`` attribute may now be ``null`` inside event series. The ``allow_multiple_entries``,
``allow_entry_after_exit``, and ``rules`` attributes have been added.
.. versionchanged:: 3.11
The ``subevent_match`` and ``exclude`` query parameters have been added.
.. versionchanged:: 3.12
The ``exit_all_at`` attribute has been added.
.. versionchanged:: 3.17
The ``ends_after`` and ``expand`` query parameters have been added.
.. versionchanged:: 4.12 .. versionchanged:: 4.12
The ``addon_match`` attribute has been added. The ``addon_match`` attribute has been added.
@@ -98,8 +115,6 @@ Endpoints
:query string ends_after: Exclude all check-in lists attached to a sub-event that is already in the past at the given time. :query string ends_after: Exclude all check-in lists attached to a sub-event that is already in the past at the given time.
:query string expand: Expand a field into a full object. Currently only ``subevent`` is supported. Can be passed multiple times. :query string expand: Expand a field into a full object. Currently only ``subevent`` is supported. Can be passed multiple times.
:query string exclude: Exclude a field from the output, e.g. ``checkin_count``. Can be used as a performance optimization. Can be passed multiple times. :query string exclude: Exclude a field from the output, e.g. ``checkin_count``. Can be used as a performance optimization. Can be passed multiple times.
:query string ordering: Manually set the ordering of results. Valid fields to be used are ``id``, ``name``, and ``subevent__date_from``,
Default: ``subevent__date_from,name``
:param organizer: The ``slug`` field of the organizer to fetch :param organizer: The ``slug`` field of the organizer to fetch
:param event: The ``slug`` field of the event to fetch :param event: The ``slug`` field of the event to fetch
:statuscode 200: no error :statuscode 200: no error

View File

@@ -52,9 +52,34 @@ sales_channels list A list of sales
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.3
The attributes ``geo_lat`` and ``geo_lon`` have been added.
.. versionchanged:: 3.4
The attribute ``timezone`` has been added.
.. versionchanged:: 3.7
The attribute ``item_meta_properties`` has been added.
.. versionchanged:: 3.12
The attribute ``valid_keys`` has been added.
.. versionchanged:: 3.14
The attribute ``sales_channels`` has been added.
Endpoints Endpoints
--------- ---------
.. versionchanged:: 3.3
The events resource can now be filtered by meta data attributes.
.. versionchanged:: 4.0 .. versionchanged:: 4.0
The ``clone_from`` parameter has been added to the event creation endpoint. The ``clone_from`` parameter has been added to the event creation endpoint.
@@ -542,6 +567,10 @@ information about the properties.
.. warning:: This API is intended for advanced users. Even though we take care to validate your input, you will be .. warning:: This API is intended for advanced users. Even though we take care to validate your input, you will be
able to break your event using this API by creating situations of conflicting settings. Please take care. able to break your event using this API by creating situations of conflicting settings. Please take care.
.. versionchanged:: 3.6
Initial support for settings has been added to the API.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/settings/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/settings/
Get current values of event settings. Get current values of event settings.

View File

@@ -6,6 +6,10 @@ Data exporters
pretix and it's plugins include a number of data exporters that allow you to bulk download various data from pretix in pretix and it's plugins include a number of data exporters that allow you to bulk download various data from pretix in
different formats. This page shows you how to use these exporters through the API. different formats. This page shows you how to use these exporters through the API.
.. versionchanged:: 3.13
This feature has been added to the API.
.. warning:: .. warning::
While we consider the methods listed on this page to be a stable API, the availability and specific input field While we consider the methods listed on this page to be a stable API, the availability and specific input field

View File

@@ -40,6 +40,10 @@ text string Custom text of
Endpoints Endpoints
--------- ---------
.. versionadded:: 3.14
The transaction list endpoint was added.
.. http:get:: /api/v1/organizers/(organizer)/giftcards/ .. http:get:: /api/v1/organizers/(organizer)/giftcards/
Returns a list of all gift cards issued by a given organizer. Returns a list of all gift cards issued by a given organizer.
@@ -253,6 +257,10 @@ Endpoints
"value": "15.37" "value": "15.37"
} }
.. versionchanged:: 3.5
This endpoint now returns status code ``409`` if the transaction would lead to a negative gift card value.
:param organizer: The ``slug`` field of the organizer to modify :param organizer: The ``slug`` field of the organizer to modify
:param id: The ``id`` field of the gift card to modify :param id: The ``id`` field of the gift card to modify
:query boolean include_accepted: Also show gift cards issued by other organizers that are accepted by this organizer. :query boolean include_accepted: Also show gift cards issued by other organizers that are accepted by this organizer.

View File

@@ -108,6 +108,16 @@ internal_reference string Customer's refe
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.4
The attribute ``lines.number`` has been added.
.. versionchanged:: 3.17
The attribute ``invoice_to_*``, ``invoice_from_*``, ``custom_field``, ``lines.item``, ``lines.variation``, ``lines.event_date_from``,
``lines.event_date_to``, and ``lines.attendee_name`` have been added.
``refers`` now returns an invoice number including the prefix.
.. versionchanged:: 4.1 .. versionchanged:: 4.1
The attributes ``fee_type`` and ``fee_internal_type`` have been added. The attributes ``fee_type`` and ``fee_internal_type`` have been added.

View File

@@ -43,13 +43,8 @@ available_until datetime The last date t
hide_without_voucher boolean If ``true``, this variation is only shown during the voucher hide_without_voucher boolean If ``true``, this variation is only shown during the voucher
redemption process, but not in the normal shop redemption process, but not in the normal shop
frontend. frontend.
meta_data object Values set for event-specific meta data parameters.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 4.16
The ``meta_data`` attribute has been added.
Endpoints Endpoints
--------- ---------
@@ -99,7 +94,6 @@ Endpoints
"default_price": "223.00", "default_price": "223.00",
"price": 223.0, "price": 223.0,
"original_price": null, "original_price": null,
"meta_data": {}
}, },
{ {
"id": 3, "id": 3,
@@ -114,8 +108,7 @@ Endpoints
"description": {}, "description": {},
"position": 1, "position": 1,
"default_price": null, "default_price": null,
"price": 15.0, "price": 15.0
"meta_data": {}
} }
] ]
} }
@@ -168,8 +161,7 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"position": 0, "position": 0
"meta_data": {}
} }
:param organizer: The ``slug`` field of the organizer to fetch :param organizer: The ``slug`` field of the organizer to fetch
@@ -206,8 +198,7 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"position": 0, "position": 0
"meta_data": {}
} }
**Example response**: **Example response**:
@@ -234,8 +225,7 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"position": 0, "position": 0
"meta_data": {}
} }
:param organizer: The ``slug`` field of the organizer of the event/item to create a variation for :param organizer: The ``slug`` field of the organizer of the event/item to create a variation for
@@ -293,8 +283,7 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"position": 1, "position": 1
"meta_data": {}
} }
:param organizer: The ``slug`` field of the organizer to modify :param organizer: The ``slug`` field of the organizer to modify

View File

@@ -35,12 +35,6 @@ tax_rule integer The internal ID
admission boolean ``true`` for items that grant admission to the event admission boolean ``true`` for items that grant admission to the event
(such as primary tickets) and ``false`` for others (such as primary tickets) and ``false`` for others
(such as add-ons or merchandise). (such as add-ons or merchandise).
personalized boolean ``true`` for items that require personalization according
to event settings. Only affects system-level fields, not
custom questions. Currently only allowed for products with
``admission`` set to ``true``. For backwards compatibility,
when creating new items and this field is not given, it defaults
to the same value as ``admission``.
position integer An integer, used for sorting position integer An integer, used for sorting
picture file A product picture to be displayed in the shop picture file A product picture to be displayed in the shop
(can be ``null``). (can be ``null``).
@@ -129,7 +123,6 @@ variations list of objects A list with one
├ hide_without_voucher boolean If ``true``, this variation is only shown during the voucher ├ hide_without_voucher boolean If ``true``, this variation is only shown during the voucher
redemption process, but not in the normal shop redemption process, but not in the normal shop
frontend. frontend.
├ meta_data object Values set for event-specific meta data parameters.
└ position integer An integer, used for sorting └ position integer An integer, used for sorting
addons list of objects Definition of add-ons that can be chosen for this item. addons list of objects Definition of add-ons that can be chosen for this item.
Only writable during creation, Only writable during creation,
@@ -153,6 +146,14 @@ bundles list of objects Definition of b
meta_data object Values set for event-specific meta data parameters. meta_data object Values set for event-specific meta data parameters.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.7
The attribute ``meta_data`` has been added.
.. versionchanged:: 3.10
The attribute ``multi_allowed`` has been added to ``addons``.
.. versionchanged:: 4.0 .. versionchanged:: 4.0
The attributes ``require_membership``, ``require_membership_types``, ``grant_membership_type``, ``grant_membership_duration_like_event``, The attributes ``require_membership``, ``require_membership_types``, ``grant_membership_type``, ``grant_membership_duration_like_event``,
@@ -162,10 +163,6 @@ meta_data object Values set for
The attributes ``require_membership_hidden`` attribute has been added. The attributes ``require_membership_hidden`` attribute has been added.
.. versionchanged:: 4.16
The ``variations[x].meta_data`` attribute has been added. The ``personalized`` attribute has been added.
Notes Notes
----- -----
@@ -219,7 +216,6 @@ Endpoints
"tax_rate": "0.00", "tax_rate": "0.00",
"tax_rule": 1, "tax_rule": 1,
"admission": false, "admission": false,
"personalized": false,
"issue_giftcard": false, "issue_giftcard": false,
"meta_data": {}, "meta_data": {},
"position": 0, "position": 0,
@@ -259,7 +255,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 0 "position": 0
}, },
{ {
@@ -275,7 +270,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 1 "position": 1
} }
], ],
@@ -336,7 +330,6 @@ Endpoints
"tax_rate": "0.00", "tax_rate": "0.00",
"tax_rule": 1, "tax_rule": 1,
"admission": false, "admission": false,
"personalized": false,
"issue_giftcard": false, "issue_giftcard": false,
"meta_data": {}, "meta_data": {},
"position": 0, "position": 0,
@@ -376,7 +369,6 @@ Endpoints
"available_from": null, "available_from": null,
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"meta_data": {},
"position": 0 "position": 0
}, },
{ {
@@ -392,7 +384,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 1 "position": 1
} }
], ],
@@ -434,7 +425,6 @@ Endpoints
"tax_rate": "0.00", "tax_rate": "0.00",
"tax_rule": 1, "tax_rule": 1,
"admission": false, "admission": false,
"personalized": false,
"issue_giftcard": false, "issue_giftcard": false,
"meta_data": {}, "meta_data": {},
"position": 0, "position": 0,
@@ -473,7 +463,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 0 "position": 0
}, },
{ {
@@ -489,7 +478,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 1 "position": 1
} }
], ],
@@ -519,7 +507,6 @@ Endpoints
"tax_rate": "0.00", "tax_rate": "0.00",
"tax_rule": 1, "tax_rule": 1,
"admission": false, "admission": false,
"personalized": false,
"issue_giftcard": false, "issue_giftcard": false,
"meta_data": {}, "meta_data": {},
"position": 0, "position": 0,
@@ -559,7 +546,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 0 "position": 0
}, },
{ {
@@ -575,7 +561,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 1 "position": 1
} }
], ],
@@ -636,7 +621,6 @@ Endpoints
"tax_rate": "0.00", "tax_rate": "0.00",
"tax_rule": 1, "tax_rule": 1,
"admission": false, "admission": false,
"personalized": false,
"issue_giftcard": false, "issue_giftcard": false,
"meta_data": {}, "meta_data": {},
"position": 0, "position": 0,
@@ -676,7 +660,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 0 "position": 0
}, },
{ {
@@ -692,7 +675,6 @@ Endpoints
"available_until": null, "available_until": null,
"hide_without_voucher": false, "hide_without_voucher": false,
"description": null, "description": null,
"meta_data": {},
"position": 1 "position": 1
} }
], ],

View File

@@ -98,6 +98,30 @@ last_modified datetime Last modificati
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.5
The ``order.fees.canceled`` attribute has been added.
.. versionchanged:: 3.8
The ``reactivate`` operation has been added.
.. versionchanged:: 3.10
The ``search`` query parameter has been added.
.. versionchanged:: 3.11
The ``exclude`` and ``subevent_after`` query parameter has been added.
.. versionchanged:: 3.13
The ``subevent_before`` query parameter has been added.
.. versionchanged:: 3.14
The ``phone`` attribute has been added.
.. versionchanged:: 4.0 .. versionchanged:: 4.0
The ``customer`` attribute has been added. The ``customer`` attribute has been added.
@@ -118,10 +142,6 @@ last_modified datetime Last modificati
The ``order.fees.id`` attribute has been added. The ``order.fees.id`` attribute has been added.
.. versionchanged:: 4.15
The ``include`` query parameter has been added.
.. _order-position-resource: .. _order-position-resource:
@@ -158,7 +178,6 @@ tax_rule integer The ID of the u
secret string Secret code printed on the tickets for validation secret string Secret code printed on the tickets for validation
addon_to integer Internal ID of the position this position is an add-on for (or ``null``) addon_to integer Internal ID of the position this position is an add-on for (or ``null``)
subevent integer ID of the date inside an event series this position belongs to (or ``null``). subevent integer ID of the date inside an event series this position belongs to (or ``null``).
discount integer ID of a discount that has been used during the creation of this position in some way (or ``null``).
pseudonymization_id string A random ID, e.g. for use in lead scanning apps pseudonymization_id string A random ID, e.g. for use in lead scanning apps
checkins list of objects List of **successful** check-ins with this ticket checkins list of objects List of **successful** check-ins with this ticket
├ id integer Internal ID of the check-in event ├ id integer Internal ID of the check-in event
@@ -186,6 +205,27 @@ pdf_data object Data object req
``pdf_data=true`` query parameter to your request. ``pdf_data=true`` query parameter to your request.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.3
The ``url`` of a ticket ``download`` can now also return a ``text/uri-list`` instead of a file. See
:ref:`order-position-ticket-download` for details.
.. versionchanged:: 3.5
The attribute ``canceled`` has been added.
.. versionchanged:: 3.8
The attributes ``company``, ``street``, ``zipcode``, ``city``, ``country``, and ``state`` have been added.
.. versionchanged:: 3.9
The ``checkin.type`` attribute has been added.
.. versionchanged:: 3.16
Answers to file questions are now returned as an URL.
.. _order-payment-resource: .. _order-payment-resource:
Order payment resource Order payment resource
@@ -232,20 +272,15 @@ created datetime Date and time o
comment string Reason for refund (shown to the customer in some cases, can be ``null``). comment string Reason for refund (shown to the customer in some cases, can be ``null``).
execution_date datetime Date and time of completion of this refund (or ``null``) execution_date datetime Date and time of completion of this refund (or ``null``)
provider string Identification string of the payment provider provider string Identification string of the payment provider
details object Refund-specific information. This is a dictionary
with various fields that can be different between
payment providers, versions, payment states, etc. If
you read this field, you always need to be able to
deal with situations where values that you expect are
missing. Mostly, the field contains various IDs that
can be used for matching with other systems. If a
payment provider does not implement this feature,
the object is empty.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
List of all orders List of all orders
------------------ ------------------
.. versionchanged:: 3.5
The ``include_canceled_positions`` and ``include_canceled_fees`` query parameters have been added.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/
Returns a list of all orders within a given event. Returns a list of all orders within a given event.
@@ -336,7 +371,6 @@ List of all orders
"secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w", "secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w",
"addon_to": null, "addon_to": null,
"subevent": null, "subevent": null,
"discount": null,
"pseudonymization_id": "MQLJvANO3B", "pseudonymization_id": "MQLJvANO3B",
"seat": null, "seat": null,
"checkins": [ "checkins": [
@@ -413,7 +447,6 @@ List of all orders
:query datetime subevent_after: Only return orders that contain a ticket for a subevent taking place after the given date. This is an exclusive after, and it considers the **end** of the subevent (or its start, if the end is not set). :query datetime subevent_after: Only return orders that contain a ticket for a subevent taking place after the given date. This is an exclusive after, and it considers the **end** of the subevent (or its start, if the end is not set).
:query datetime subevent_before: Only return orders that contain a ticket for a subevent taking place after the given date. This is an exclusive before, and it considers the **start** of the subevent. :query datetime subevent_before: Only return orders that contain a ticket for a subevent taking place after the given date. This is an exclusive before, and it considers the **start** of the subevent.
:query string exclude: Exclude a field from the output, e.g. ``fees`` or ``positions.downloads``. Can be used as a performance optimization. Can be passed multiple times. :query string exclude: Exclude a field from the output, e.g. ``fees`` or ``positions.downloads``. Can be used as a performance optimization. Can be passed multiple times.
:query string include: Include only the given field in the output, e.g. ``fees`` or ``positions.downloads``. Can be used as a performance optimization. Can be passed multiple times. ``include`` is applied before ``exclude``, so ``exclude`` takes precedence.
:param organizer: The ``slug`` field of the organizer to fetch :param organizer: The ``slug`` field of the organizer to fetch
:param event: The ``slug`` field of the event to fetch :param event: The ``slug`` field of the event to fetch
:resheader X-Page-Generated: The server time at the beginning of the operation. If you're using this API to fetch :resheader X-Page-Generated: The server time at the beginning of the operation. If you're using this API to fetch
@@ -425,6 +458,10 @@ List of all orders
Fetching individual orders Fetching individual orders
-------------------------- --------------------------
.. versionchanged:: 3.5
The ``include_canceled_positions`` and ``include_canceled_fees`` query parameters have been added.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/
Returns information on one order, identified by its order code. Returns information on one order, identified by its order code.
@@ -509,7 +546,6 @@ Fetching individual orders
"secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w", "secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w",
"addon_to": null, "addon_to": null,
"subevent": null, "subevent": null,
"discount": null,
"pseudonymization_id": "MQLJvANO3B", "pseudonymization_id": "MQLJvANO3B",
"seat": null, "seat": null,
"checkins": [ "checkins": [
@@ -999,6 +1035,10 @@ Creating orders
Order state operations Order state operations
---------------------- ----------------------
.. versionchanged:: 3.12
The ``mark_paid`` operation now takes a ``send_email`` parameter.
.. http:post:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/mark_paid/ .. http:post:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/mark_paid/
Marks a pending or expired order as successfully paid. Marks a pending or expired order as successfully paid.
@@ -1400,6 +1440,10 @@ Sending e-mails
List of all order positions List of all order positions
--------------------------- ---------------------------
.. versionchanged:: 3.5
The ``include_canceled_positions`` and ``include_canceled_fees`` query parameters have been added.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/orderpositions/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/orderpositions/
Returns a list of all order positions within a given event. Returns a list of all order positions within a given event.
@@ -1443,7 +1487,6 @@ List of all order positions
"tax_rule": null, "tax_rule": null,
"tax_value": "0.00", "tax_value": "0.00",
"secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w", "secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w",
"discount": null,
"pseudonymization_id": "MQLJvANO3B", "pseudonymization_id": "MQLJvANO3B",
"seat": null, "seat": null,
"addon_to": null, "addon_to": null,
@@ -1554,7 +1597,6 @@ Fetching individual positions
"secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w", "secret": "z3fsn8jyufm5kpk768q69gkbyr5f4h6w",
"addon_to": null, "addon_to": null,
"subevent": null, "subevent": null,
"discount": null,
"pseudonymization_id": "MQLJvANO3B", "pseudonymization_id": "MQLJvANO3B",
"seat": null, "seat": null,
"checkins": [ "checkins": [
@@ -1654,6 +1696,10 @@ Order position ticket download
Manipulating individual positions Manipulating individual positions
--------------------------------- ---------------------------------
.. versionchanged:: 3.15
The ``PATCH`` method has been added for individual positions.
.. versionchanged:: 4.8 .. versionchanged:: 4.8
The ``PATCH`` method now supports changing items, variations, subevents, seats, prices, and tax rules. The ``PATCH`` method now supports changing items, variations, subevents, seats, prices, and tax rules.
@@ -1960,6 +2006,14 @@ otherwise, such as splitting an order or changing fees.
Order payment endpoints Order payment endpoints
----------------------- -----------------------
.. versionchanged:: 3.6
Payments can now be created through the API.
.. versionchanged:: 3.12
The ``confirm`` operation now takes a ``send_email`` parameter.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/payments/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/orders/(code)/payments/
Returns a list of all payments for an order. Returns a list of all payments for an order.
@@ -2265,7 +2319,6 @@ Order refund endpoints
"created": "2017-12-01T10:00:00Z", "created": "2017-12-01T10:00:00Z",
"execution_date": "2017-12-04T12:13:12Z", "execution_date": "2017-12-04T12:13:12Z",
"comment": "Cancellation", "comment": "Cancellation",
"details": {},
"provider": "banktransfer" "provider": "banktransfer"
} }
] ]
@@ -2309,7 +2362,6 @@ Order refund endpoints
"created": "2017-12-01T10:00:00Z", "created": "2017-12-01T10:00:00Z",
"execution_date": "2017-12-04T12:13:12Z", "execution_date": "2017-12-04T12:13:12Z",
"comment": "Cancellation", "comment": "Cancellation",
"details": {},
"provider": "banktransfer" "provider": "banktransfer"
} }
@@ -2367,7 +2419,6 @@ Order refund endpoints
"created": "2017-12-01T10:00:00Z", "created": "2017-12-01T10:00:00Z",
"execution_date": null, "execution_date": null,
"comment": "Cancellation", "comment": "Cancellation",
"details": {},
"provider": "manual" "provider": "manual"
} }
@@ -2497,6 +2548,10 @@ Revoked ticket secrets
With some non-default ticket secret generation methods, a list of revoked ticket secrets is required for proper validation. With some non-default ticket secret generation methods, a list of revoked ticket secrets is required for proper validation.
.. versionchanged:: 3.12
Added revocation lists.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/revokedsecrets/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/revokedsecrets/
Returns a list of all revoked secrets within a given event. Returns a list of all revoked secrets within a given event.

View File

@@ -109,6 +109,10 @@ information about the properties.
.. warning:: This API is intended for advanced users. Even though we take care to validate your input, you will be .. warning:: This API is intended for advanced users. Even though we take care to validate your input, you will be
able to break your shops using this API by creating situations of conflicting settings. Please take care. able to break your shops using this API by creating situations of conflicting settings. Please take care.
.. versionchanged:: 3.14
Initial support for settings has been added to the API.
.. http:get:: /api/v1/organizers/(organizer)/settings/ .. http:get:: /api/v1/organizers/(organizer)/settings/
Get current values of organizer settings. Get current values of organizer settings.

View File

@@ -76,9 +76,26 @@ dependency_value string An old version
for one value. **Deprecated.** for one value. **Deprecated.**
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.5
The attribute ``help_text`` has been added.
.. versionchanged:: 3.14
The attributes ``valid_*`` have been added.
.. versionchanged:: 3.18
The attribute ``valid_file_portrait`` have been added.
Endpoints Endpoints
--------- ---------
.. versionchanged:: 1.15
The questions endpoint has been extended by the filter queries ``ask_during_checkin``, ``requred``, and
``identifier``.
.. http:get:: /api/v1/organizers/(organizer)/events/(event)/questions/ .. http:get:: /api/v1/organizers/(organizer)/events/(event)/questions/
Returns a list of all questions within a given event. Returns a list of all questions within a given event.

View File

@@ -36,6 +36,10 @@ available_number integer Number of avail
slightly out of date. ``null`` means unlimited. slightly out of date. ``null`` means unlimited.
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.10
The attribute ``release_after_exit`` has been added.
.. versionchanged:: 4.1 .. versionchanged:: 4.1
The ``with_availability`` query parameter has been added. The ``with_availability`` query parameter has been added.

View File

@@ -59,13 +59,29 @@ seat_category_mapping object An object mappi
last_modified datetime Last modification of this object last_modified datetime Last modification of this object
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 4.15 .. versionchanged:: 3.3
The ``search`` query parameter has been added to filter sub-events by their name or location in any language. The attributes ``geo_lat`` and ``geo_lon`` have been added.
.. versionchanged:: 3.10
The ``disabled`` attribute has been added to ``item_price_overrides`` and ``variation_price_overrides``.
.. versionchanged:: 3.12
The ``last_modified`` attribute has been added.
.. versionchanged:: 3.18
The ``available_from``/``available_until`` attributes have been added to ``item_price_overrides`` and ``variation_price_overrides``.
Endpoints Endpoints
--------- ---------
.. versionchanged:: 3.3
The sub-events resource can now be filtered by meta data attributes.
.. versionchanged:: 4.1 .. versionchanged:: 4.1
The ``with_availability_for`` parameter has been added. The ``with_availability_for`` parameter has been added.
@@ -131,7 +147,6 @@ Endpoints
:query is_future: If set to ``true`` (``false``), only events that happen currently or in the future are (not) returned. :query is_future: If set to ``true`` (``false``), only events that happen currently or in the future are (not) returned.
:query is_past: If set to ``true`` (``false``), only events that are over are (not) returned. :query is_past: If set to ``true`` (``false``), only events that are over are (not) returned.
:query ends_after: If set to a date and time, only events that happen during of after the given time are returned. :query ends_after: If set to a date and time, only events that happen during of after the given time are returned.
:query search: Only return events matching a given search query.
:param organizer: The ``slug`` field of a valid organizer :param organizer: The ``slug`` field of a valid organizer
:param event: The ``slug`` field of the main event :param event: The ``slug`` field of the main event
:query datetime modified_since: Only return objects that have changed since the given date. Be careful: This does not :query datetime modified_since: Only return objects that have changed since the given date. Be careful: This does not

View File

@@ -19,8 +19,6 @@ max_usages integer The maximum num
redeemed (default: 1). redeemed (default: 1).
redeemed integer The number of times this voucher already has been redeemed integer The number of times this voucher already has been
redeemed. redeemed.
min_usages integer The minimum number of times this voucher must be
redeemed on first usage (default: 1).
valid_until datetime The voucher expiration date (or ``null``). valid_until datetime The voucher expiration date (or ``null``).
block_quota boolean If ``true``, quota is blocked for this voucher. block_quota boolean If ``true``, quota is blocked for this voucher.
allow_ignore_quota boolean If ``true``, this voucher can be redeemed even if a allow_ignore_quota boolean If ``true``, this voucher can be redeemed even if a
@@ -50,6 +48,10 @@ show_hidden_items boolean Only if set to
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 3.4
The attribute ``seat`` has been added.
Endpoints Endpoints
--------- ---------

View File

@@ -30,6 +30,12 @@ subevent integer ID of the date
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
.. versionchanged:: 1.15
The write operations ``POST``, ``PATCH``, ``PUT``, and ``DELETE`` have been added as well as a method to send out
vouchers.
Endpoints Endpoints
--------- ---------

View File

@@ -76,10 +76,6 @@ The exporter class
This is an abstract attribute, you **must** override this! This is an abstract attribute, you **must** override this!
.. autoattribute:: description
.. autoattribute:: category
.. autoattribute:: export_form_fields .. autoattribute:: export_form_fields
.. automethod:: render .. automethod:: render

View File

@@ -38,7 +38,7 @@ Frontend
.. automodule:: pretix.presale.signals .. automodule:: pretix.presale.signals
:members: order_info, order_info_top, order_meta_from_request :members: order_info, order_info_top, order_meta_from_request, order_source_from_request
Request flow Request flow
"""""""""""" """"""""""""

View File

@@ -126,8 +126,6 @@ The provider class
.. automethod:: api_payment_details .. automethod:: api_payment_details
.. automethod:: api_refund_details
.. automethod:: matching_id .. automethod:: matching_id
.. automethod:: shred_payment_info .. automethod:: shred_payment_info
@@ -138,10 +136,6 @@ The provider class
.. autoattribute:: is_meta .. autoattribute:: is_meta
.. autoattribute:: execute_payment_needs_user
.. autoattribute:: multi_use_supported
.. autoattribute:: test_mode_message .. autoattribute:: test_mode_message
.. autoattribute:: requires_invoice_immediately .. autoattribute:: requires_invoice_immediately

View File

@@ -184,6 +184,11 @@ Most of these methods work identically on :class:`pretix.base.models.TeamAPIToke
Staff sessions Staff sessions
-------------- --------------
.. versionchanged:: 1.14
In 1.14, the ``User.is_superuser`` attribute has been deprecated and statically set to return ``False``. Staff
sessions have been newly introduced.
System administrators of a pretix instance are identified by the ``is_staff`` attribute on the user model. By default, System administrators of a pretix instance are identified by the ``is_staff`` attribute on the user model. By default,
the regular permission rules apply for users with ``is_staff = True``. The only difference is that such users can the regular permission rules apply for users with ``is_staff = True``. The only difference is that such users can
temporarily turn on "staff mode" via a button in the user interface that grants them **all permissions** as long as temporarily turn on "staff mode" via a button in the user interface that grants them **all permissions** as long as

View File

@@ -91,7 +91,6 @@ Field Type Description
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
id integer Internal content ID id integer Internal content ID
title multi-lingual string The content title (required) title multi-lingual string The content title (required)
internal_name string An optional name that is only used in the backend
content_type string The type of content, valid values are ``webinar``, ``video``, ``livestream``, ``link``, ``file`` content_type string The type of content, valid values are ``webinar``, ``video``, ``livestream``, ``link``, ``file``
url string The location of the digital content url string The location of the digital content
file file A downloadable file. Either ``url`` or ``file`` must be ``null``. file file A downloadable file. Either ``url`` or ``file`` must be ``null``.

View File

@@ -17,13 +17,9 @@ Field Type Description
id integer Internal layout ID id integer Internal layout ID
name string Internal layout description name string Internal layout description
default boolean ``true`` if this is the default layout default boolean ``true`` if this is the default layout
layout list Dynamic layout specification. Each list element layout object Layout specification for libpretixprint
corresponds to one dynamic element of the layout.
The current version of the schema in use can be found
`here`_.
Submitting invalid content can lead to application errors.
background URL Background PDF file background URL Background PDF file
item_assignments list of objects Products this layout is assigned to (currently read-only) item_assignments list of objects Products this layout is assigned to
├ sales_channel string Sales channel (defaults to ``web``). ├ sales_channel string Sales channel (defaults to ``web``).
└ item integer Item ID └ item integer Item ID
===================================== ========================== ======================================================= ===================================== ========================== =======================================================
@@ -62,7 +58,7 @@ Endpoints
"name": "Default layout", "name": "Default layout",
"default": true, "default": true,
"layout": {…}, "layout": {…},
"background": null, "background": {},
"item_assignments": [] "item_assignments": []
} }
] ]
@@ -100,7 +96,7 @@ Endpoints
"name": "Default layout", "name": "Default layout",
"default": true, "default": true,
"layout": {…}, "layout": {…},
"background": null, "background": {},
"item_assignments": [] "item_assignments": []
} }
@@ -151,122 +147,3 @@ Endpoints
:statuscode 200: no error :statuscode 200: no error
:statuscode 401: Authentication failure :statuscode 401: Authentication failure
:statuscode 403: The requested organizer does not exist **or** you have no permission to view it. :statuscode 403: The requested organizer does not exist **or** you have no permission to view it.
.. http:post:: /api/v1/organizers/(organizer)/events/(event)/ticketlayouts/
Creates a new ticket layout
**Example request**:
.. sourcecode:: http
POST /api/v1/organizers/bigevents/events/sampleconf/ticketlayouts/ HTTP/1.1
Host: pretix.eu
Accept: application/json, text/javascript
Content-Type: application/json
{
"name": "Default layout",
"default": true,
"layout": […],
"background": null,
"item_assignments": []
}
**Example response**:
.. sourcecode:: http
HTTP/1.1 201 Created
Vary: Accept
Content-Type: application/json
{
"id": 1,
"name": "Default layout",
"default": true,
"layout": […],
"background": null,
"item_assignments": []
}
:param organizer: The ``slug`` field of the organizer of the event to create a layout for
:param event: The ``slug`` field of the event to create a layout for
:statuscode 201: no error
:statuscode 400: The layout could not be created due to invalid submitted data.
:statuscode 401: Authentication failure
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to create this resource.
.. http:patch:: /api/v1/organizers/(organizer)/events/(event)/ticketlayouts/(id)/
Update a layout. You can also use ``PUT`` instead of ``PATCH``. With ``PUT``, you have to provide all fields of
the resource, other fields will be reset to default. With ``PATCH``, you only need to provide the fields that you
want to change.
**Example request**:
.. sourcecode:: http
PATCH /api/v1/organizers/bigevents/events/sampleconf/ticketlayouts/1/ HTTP/1.1
Host: pretix.eu
Accept: application/json, text/javascript
Content-Type: application/json
Content-Length: 94
{
"name": "Default layout"
}
**Example response**:
.. sourcecode:: http
HTTP/1.1 200 OK
Vary: Accept
Content-Type: application/json
{
"id": 1,
"name": "Default layout",
"default": true,
"layout": […],
"background": null,
"item_assignments": []
}
:param organizer: The ``slug`` field of the organizer to modify
:param event: The ``slug`` field of the event to modify
:param id: The ``id`` field of the layout to modify
:statuscode 200: no error
:statuscode 400: The layout could not be modified due to invalid submitted data
:statuscode 401: Authentication failure
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to change this resource.
.. http:delete:: /api/v1/organizers/(organizer)/events/(event)/ticketlayouts/(id)/
Delete a layout.
**Example request**:
.. sourcecode:: http
DELETE /api/v1/organizers/bigevents/events/sampleconf/ticketlayouts/1/ HTTP/1.1
Host: pretix.eu
Accept: application/json, text/javascript
**Example response**:
.. sourcecode:: http
HTTP/1.1 204 No Content
Vary: Accept
:param organizer: The ``slug`` field of the organizer to modify
:param event: The ``slug`` field of the event to modify
:param id: The ``id`` field of the layout to delete
:statuscode 204: no error
:statuscode 401: Authentication failure
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to delete this resource.
.. _here: https://github.com/pretix/pretix/blob/master/src/pretix/static/schema/pdf-layout.schema.json

View File

@@ -97,7 +97,6 @@ overpayment
param param
passphrase passphrase
percental percental
personalization
pluggable pluggable
positionid positionid
pre pre

View File

@@ -447,4 +447,8 @@ Hosted or pretix Enterprise are active, you can pass the following fields:
</script> </script>
.. versionchanged:: 3.6
Dynamically opening the widget has been added in pretix 3.6.
.. _Let's Encrypt: https://letsencrypt.org/ .. _Let's Encrypt: https://letsencrypt.org/

View File

@@ -19,4 +19,4 @@
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see # You should have received a copy of the GNU Affero General Public License along with this program. If not, see
# <https://www.gnu.org/licenses/>. # <https://www.gnu.org/licenses/>.
# #
__version__ = "4.16.1" __version__ = "4.14.1.dev0"

View File

@@ -196,7 +196,6 @@ class PretixPosSecurityProfile(AllowListSecurityProfile):
('POST', 'plugins:pretix_posbackend:posdebuglogentry-bulk-create'), ('POST', 'plugins:pretix_posbackend:posdebuglogentry-bulk-create'),
('GET', 'plugins:pretix_posbackend:poscashier-list'), ('GET', 'plugins:pretix_posbackend:poscashier-list'),
('POST', 'plugins:pretix_posbackend:stripeterminal.token'), ('POST', 'plugins:pretix_posbackend:stripeterminal.token'),
('POST', 'plugins:pretix_posbackend:stripeterminal.paymentintent'),
('PUT', 'plugins:pretix_posbackend:file.upload'), ('PUT', 'plugins:pretix_posbackend:file.upload'),
('GET', 'api-v1:revokedsecrets-list'), ('GET', 'api-v1:revokedsecrets-list'),
('GET', 'api-v1:event.settings'), ('GET', 'api-v1:event.settings'),

View File

@@ -19,17 +19,11 @@
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see # You should have received a copy of the GNU Affero General Public License along with this program. If not, see
# <https://www.gnu.org/licenses/>. # <https://www.gnu.org/licenses/>.
# #
import logging
import ujson
from rest_framework import exceptions
from rest_framework.response import Response from rest_framework.response import Response
from rest_framework.views import exception_handler, status from rest_framework.views import exception_handler, status
from pretix.base.services.locking import LockTimeoutException from pretix.base.services.locking import LockTimeoutException
logger = logging.getLogger(__name__)
def custom_exception_handler(exc, context): def custom_exception_handler(exc, context):
response = exception_handler(exc, context) response = exception_handler(exc, context)
@@ -43,7 +37,4 @@ def custom_exception_handler(exc, context):
} }
) )
if isinstance(exc, exceptions.APIException):
logger.info(f'API Exception [{exc.status_code}]: {ujson.dumps(exc.detail)}')
return response return response

View File

@@ -32,7 +32,6 @@ from rest_framework import status
from pretix.api.models import ApiCall from pretix.api.models import ApiCall
from pretix.base.models import Organizer from pretix.base.models import Organizer
from pretix.helpers import OF_SELF
class IdempotencyMiddleware: class IdempotencyMiddleware:
@@ -57,7 +56,7 @@ class IdempotencyMiddleware:
idempotency_key = request.headers.get('X-Idempotency-Key', '') idempotency_key = request.headers.get('X-Idempotency-Key', '')
with transaction.atomic(): with transaction.atomic():
call, created = ApiCall.objects.select_for_update(of=OF_SELF).get_or_create( call, created = ApiCall.objects.select_for_update().get_or_create(
auth_hash=auth_hash, auth_hash=auth_hash,
idempotency_key=idempotency_key, idempotency_key=idempotency_key,
defaults={ defaults={

View File

@@ -1,77 +0,0 @@
# Generated by Django 3.2.16 on 2022-12-17 18:47
import uuid
import django.db.models.deletion
import oauth2_provider.generators
import oauth2_provider.models
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0226_itemvariationmetavalue'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
('pretixapi', '0008_webhookcallretry'),
]
run_before = [
('oauth2_provider', '0002_auto_20190406_1805'),
]
operations = [
migrations.AddField(
model_name='oauthapplication',
name='algorithm',
field=models.CharField(default='', max_length=5),
),
migrations.AddField(
model_name='oauthgrant',
name='claims',
field=models.TextField(default=''),
preserve_default=False,
),
migrations.AddField(
model_name='oauthgrant',
name='code_challenge',
field=models.CharField(default='', max_length=128),
),
migrations.AddField(
model_name='oauthgrant',
name='code_challenge_method',
field=models.CharField(default='', max_length=10),
),
migrations.AddField(
model_name='oauthgrant',
name='nonce',
field=models.CharField(default='', max_length=255),
),
migrations.AlterField(
model_name='oauthapplication',
name='client_secret',
field=oauth2_provider.models.ClientSecretField(db_index=True, default=oauth2_provider.generators.generate_client_secret, max_length=255),
),
migrations.CreateModel(
name='OAuthIDToken',
fields=[
('id', models.BigAutoField(primary_key=True, serialize=False)),
('jti', models.UUIDField(default=uuid.uuid4, unique=True)),
('expires', models.DateTimeField()),
('scope', models.TextField()),
('created', models.DateTimeField(auto_now_add=True)),
('updated', models.DateTimeField(auto_now=True)),
('application', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.OAUTH2_PROVIDER_APPLICATION_MODEL)),
('organizers', models.ManyToManyField(to='pretixbase.Organizer')),
('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='pretixapi_oauthidtoken', to=settings.AUTH_USER_MODEL)),
],
options={
'abstract': False,
},
),
migrations.AddField(
model_name='oauthaccesstoken',
name='id_token',
field=models.OneToOneField(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='access_token', to='pretixapi.oauthidtoken'),
),
]

View File

@@ -29,8 +29,8 @@ from oauth2_provider.generators import (
generate_client_id, generate_client_secret, generate_client_id, generate_client_secret,
) )
from oauth2_provider.models import ( from oauth2_provider.models import (
AbstractAccessToken, AbstractApplication, AbstractGrant, AbstractIDToken, AbstractAccessToken, AbstractApplication, AbstractGrant,
AbstractRefreshToken, ClientSecretField, AbstractRefreshToken,
) )
from oauth2_provider.validators import URIValidator from oauth2_provider.validators import URIValidator
@@ -46,7 +46,7 @@ class OAuthApplication(AbstractApplication):
verbose_name=_("Client ID"), verbose_name=_("Client ID"),
max_length=100, unique=True, default=generate_client_id, db_index=True max_length=100, unique=True, default=generate_client_id, db_index=True
) )
client_secret = ClientSecretField( client_secret = models.CharField(
verbose_name=_("Client secret"), verbose_name=_("Client secret"),
max_length=255, blank=False, default=generate_client_secret, db_index=True max_length=255, blank=False, default=generate_client_secret, db_index=True
) )
@@ -67,26 +67,12 @@ class OAuthGrant(AbstractGrant):
redirect_uri = models.CharField(max_length=2500) # Only 255 in AbstractGrant, which caused problems redirect_uri = models.CharField(max_length=2500) # Only 255 in AbstractGrant, which caused problems
class OAuthIDToken(AbstractIDToken):
application = models.ForeignKey(
OAuthApplication, on_delete=models.CASCADE,
)
organizers = models.ManyToManyField('pretixbase.Organizer')
class OAuthAccessToken(AbstractAccessToken): class OAuthAccessToken(AbstractAccessToken):
source_refresh_token = models.OneToOneField( source_refresh_token = models.OneToOneField(
# unique=True implied by the OneToOneField # unique=True implied by the OneToOneField
'OAuthRefreshToken', on_delete=models.SET_NULL, blank=True, null=True, 'OAuthRefreshToken', on_delete=models.SET_NULL, blank=True, null=True,
related_name="refreshed_access_token" related_name="refreshed_access_token"
) )
id_token = models.OneToOneField(
OAuthIDToken,
on_delete=models.CASCADE,
blank=True,
null=True,
related_name="access_token",
)
application = models.ForeignKey( application = models.ForeignKey(
OAuthApplication, on_delete=models.CASCADE, blank=True, null=True, OAuthApplication, on_delete=models.CASCADE, blank=True, null=True,
) )

View File

@@ -237,14 +237,12 @@ class CartPositionCreateSerializer(BaseCartPositionCreateSerializer):
for addon_data in addons_data: for addon_data in addons_data:
addon_data['addon_to'] = cp addon_data['addon_to'] = cp
addon_data['is_bundled'] = False addon_data['is_bundled'] = False
addon_data['cart_id'] = cp.cart_id
super().create(addon_data) super().create(addon_data)
if bundled_data: if bundled_data:
for bundle_data in bundled_data: for bundle_data in bundled_data:
bundle_data['addon_to'] = cp bundle_data['addon_to'] = cp
bundle_data['is_bundled'] = True bundle_data['is_bundled'] = True
bundle_data['cart_id'] = cp.cart_id
super().create(bundle_data) super().create(bundle_data)
return cp return cp

View File

@@ -411,8 +411,7 @@ class CloneEventSerializer(EventSerializer):
has_subevents = validated_data.pop('has_subevents', None) has_subevents = validated_data.pop('has_subevents', None)
tz = validated_data.pop('timezone', None) tz = validated_data.pop('timezone', None)
sales_channels = validated_data.pop('sales_channels', None) sales_channels = validated_data.pop('sales_channels', None)
date_admission = validated_data.pop('date_admission', None) new_event = super().create(validated_data)
new_event = super().create({**validated_data, 'plugins': None})
event = Event.objects.filter(slug=self.context['event'], organizer=self.context['organizer'].pk).first() event = Event.objects.filter(slug=self.context['event'], organizer=self.context['organizer'].pk).first()
new_event.copy_data_from(event) new_event.copy_data_from(event)
@@ -427,10 +426,6 @@ class CloneEventSerializer(EventSerializer):
new_event.sales_channels = sales_channels new_event.sales_channels = sales_channels
if has_subevents is not None: if has_subevents is not None:
new_event.has_subevents = has_subevents new_event.has_subevents = has_subevents
if has_subevents is not None:
new_event.has_subevents = has_subevents
if date_admission is not None:
new_event.date_admission = date_admission
new_event.save() new_event.save()
if tz: if tz:
new_event.settings.timezone = tz new_event.settings.timezone = tz
@@ -760,9 +755,6 @@ class EventSettingsSerializer(SettingsSerializer):
'invoice_logo_image', 'invoice_logo_image',
'cancel_allow_user', 'cancel_allow_user',
'cancel_allow_user_until', 'cancel_allow_user_until',
'cancel_allow_user_unpaid_keep',
'cancel_allow_user_unpaid_keep_fees',
'cancel_allow_user_unpaid_keep_percentage',
'cancel_allow_user_paid', 'cancel_allow_user_paid',
'cancel_allow_user_paid_until', 'cancel_allow_user_paid_until',
'cancel_allow_user_paid_keep', 'cancel_allow_user_paid_keep',

View File

@@ -22,10 +22,8 @@
from django import forms from django import forms
from django.http import QueryDict from django.http import QueryDict
from rest_framework import serializers from rest_framework import serializers
from rest_framework.exceptions import ValidationError
from pretix.base.exporter import OrganizerLevelExportMixin from pretix.base.exporter import OrganizerLevelExportMixin
from pretix.base.timeframes import DateFrameField, SerializerDateFrameField
class FormFieldWrapperField(serializers.Field): class FormFieldWrapperField(serializers.Field):
@@ -144,12 +142,6 @@ class JobRunSerializer(serializers.Serializer):
allow_null=not v.required, allow_null=not v.required,
validators=v.validators, validators=v.validators,
) )
elif isinstance(v, DateFrameField):
self.fields[k] = SerializerDateFrameField(
required=v.required,
allow_null=not v.required,
validators=v.validators,
)
else: else:
self.fields[k] = FormFieldWrapperField(form_field=v, required=v.required, allow_null=not v.required) self.fields[k] = FormFieldWrapperField(form_field=v, required=v.required, allow_null=not v.required)
@@ -159,40 +151,5 @@ class JobRunSerializer(serializers.Serializer):
for k, v in self.fields.items(): for k, v in self.fields.items():
if isinstance(v, serializers.ManyRelatedField) and k not in data: if isinstance(v, serializers.ManyRelatedField) and k not in data:
data[k] = [] data[k] = []
for fk in self.fields.keys():
# Backwards compatibility for exports that used to take e.g. (date_from, date_to) or (event_date_from, event_date_to)
# and now only take date_range.
if fk.endswith("_range") and isinstance(self.fields[fk], SerializerDateFrameField) and fk not in data:
if fk.replace("_range", "_from") in data:
d_from = data.pop(fk.replace("_range", "_from"))
if d_from:
d_from = serializers.DateField().to_internal_value(d_from)
else:
d_from = None
if fk.replace("_range", "_to") in data:
d_to = data.pop(fk.replace("_range", "_to"))
if d_to:
d_to = serializers.DateField().to_internal_value(d_to)
else:
d_to = None
data[fk] = f'{d_from.isoformat() if d_from else ""}/{d_to.isoformat() if d_to else ""}'
data = super().to_internal_value(data) data = super().to_internal_value(data)
return data return data
def is_valid(self, raise_exception=False):
super().is_valid(raise_exception=raise_exception)
fields_keys = set(self.fields.keys())
input_keys = set(self.initial_data.keys())
additional_fields = input_keys - fields_keys
if bool(additional_fields):
self._errors['fields'] = ['Additional fields not allowed: {}.'.format(list(additional_fields))]
if self._errors and raise_exception:
raise ValidationError(self.errors)
return not bool(self._errors)

View File

@@ -47,14 +47,13 @@ from pretix.api.serializers.fields import UploadedFileField
from pretix.api.serializers.i18n import I18nAwareModelSerializer from pretix.api.serializers.i18n import I18nAwareModelSerializer
from pretix.base.models import ( from pretix.base.models import (
Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaValue, ItemVariation, Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaValue, ItemVariation,
ItemVariationMetaValue, Question, QuestionOption, Quota, Question, QuestionOption, Quota,
) )
class InlineItemVariationSerializer(I18nAwareModelSerializer): class InlineItemVariationSerializer(I18nAwareModelSerializer):
price = serializers.DecimalField(read_only=True, decimal_places=2, max_digits=10, price = serializers.DecimalField(read_only=True, decimal_places=2, max_digits=10,
coerce_to_string=True) coerce_to_string=True)
meta_data = MetaDataField(required=False, source='*')
class Meta: class Meta:
model = ItemVariation model = ItemVariation
@@ -62,23 +61,16 @@ class InlineItemVariationSerializer(I18nAwareModelSerializer):
'position', 'default_price', 'price', 'original_price', 'require_approval', 'position', 'default_price', 'price', 'original_price', 'require_approval',
'require_membership', 'require_membership_types', 'require_membership', 'require_membership_types',
'require_membership_hidden', 'available_from', 'available_until', 'require_membership_hidden', 'available_from', 'available_until',
'sales_channels', 'hide_without_voucher', 'meta_data') 'sales_channels', 'hide_without_voucher',)
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.fields['require_membership_types'].queryset = lazy(lambda: self.context['event'].organizer.membership_types.all(), QuerySet) self.fields['require_membership_types'].queryset = lazy(lambda: self.context['event'].organizer.membership_types.all(), QuerySet)
def validate_meta_data(self, value):
for key in value['meta_data'].keys():
if key not in self.parent.parent.item_meta_properties:
raise ValidationError(_('Item meta data property \'{name}\' does not exist.').format(name=key))
return value
class ItemVariationSerializer(I18nAwareModelSerializer): class ItemVariationSerializer(I18nAwareModelSerializer):
price = serializers.DecimalField(read_only=True, decimal_places=2, max_digits=10, price = serializers.DecimalField(read_only=True, decimal_places=2, max_digits=10,
coerce_to_string=True) coerce_to_string=True)
meta_data = MetaDataField(required=False, source='*')
class Meta: class Meta:
model = ItemVariation model = ItemVariation
@@ -86,67 +78,12 @@ class ItemVariationSerializer(I18nAwareModelSerializer):
'position', 'default_price', 'price', 'original_price', 'require_approval', 'position', 'default_price', 'price', 'original_price', 'require_approval',
'require_membership', 'require_membership_types', 'require_membership', 'require_membership_types',
'require_membership_hidden', 'available_from', 'available_until', 'require_membership_hidden', 'available_from', 'available_until',
'sales_channels', 'hide_without_voucher', 'meta_data') 'sales_channels', 'hide_without_voucher',)
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.fields['require_membership_types'].queryset = self.context['event'].organizer.membership_types.all() self.fields['require_membership_types'].queryset = self.context['event'].organizer.membership_types.all()
@transaction.atomic
def create(self, validated_data):
meta_data = validated_data.pop('meta_data', None)
require_membership_types = validated_data.pop('require_membership_types', [])
variation = ItemVariation.objects.create(**validated_data)
if require_membership_types:
variation.require_membership_types.add(*require_membership_types)
# Meta data
if meta_data is not None:
for key, value in meta_data.items():
ItemVariationMetaValue.objects.create(
property=self.item_meta_properties.get(key),
value=value,
variation=variation
)
return variation
@cached_property
def item_meta_properties(self):
return {
p.name: p for p in self.context['request'].event.item_meta_properties.all()
}
def validate_meta_data(self, value):
for key in value['meta_data'].keys():
if key not in self.item_meta_properties:
raise ValidationError(_('Item meta data property \'{name}\' does not exist.').format(name=key))
return value
def update(self, instance, validated_data):
meta_data = validated_data.pop('meta_data', None)
variation = super().update(instance, validated_data)
# Meta data
if meta_data is not None:
current = {mv.property: mv for mv in variation.meta_values.select_related('property')}
for key, value in meta_data.items():
prop = self.item_meta_properties.get(key)
if prop in current:
current[prop].value = value
current[prop].save()
else:
variation.meta_values.create(
property=self.item_meta_properties.get(key),
value=value
)
for prop, current_object in current.items():
if prop.name not in meta_data:
current_object.delete()
return variation
class InlineItemBundleSerializer(serializers.ModelSerializer): class InlineItemBundleSerializer(serializers.ModelSerializer):
class Meta: class Meta:
@@ -234,7 +171,7 @@ class ItemSerializer(I18nAwareModelSerializer):
class Meta: class Meta:
model = Item model = Item
fields = ('id', 'category', 'name', 'internal_name', 'active', 'sales_channels', 'description', fields = ('id', 'category', 'name', 'internal_name', 'active', 'sales_channels', 'description',
'default_price', 'free_price', 'tax_rate', 'tax_rule', 'admission', 'personalized', 'default_price', 'free_price', 'tax_rate', 'tax_rule', 'admission',
'position', 'picture', 'available_from', 'available_until', 'position', 'picture', 'available_from', 'available_until',
'require_voucher', 'hide_without_voucher', 'allow_cancel', 'require_bundling', 'require_voucher', 'hide_without_voucher', 'allow_cancel', 'require_bundling',
'min_per_order', 'max_per_order', 'checkin_attention', 'has_variations', 'variations', 'min_per_order', 'max_per_order', 'checkin_attention', 'has_variations', 'variations',
@@ -247,8 +184,6 @@ class ItemSerializer(I18nAwareModelSerializer):
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.fields['default_price'].allow_null = False
self.fields['default_price'].required = True
if not self.read_only: if not self.read_only:
self.fields['require_membership_types'].queryset = self.context['event'].organizer.membership_types.all() self.fields['require_membership_types'].queryset = self.context['event'].organizer.membership_types.all()
self.fields['grant_membership_type'].queryset = self.context['event'].organizer.membership_types.all() self.fields['grant_membership_type'].queryset = self.context['event'].organizer.membership_types.all()
@@ -262,15 +197,6 @@ class ItemSerializer(I18nAwareModelSerializer):
Item.clean_per_order(data.get('min_per_order'), data.get('max_per_order')) Item.clean_per_order(data.get('min_per_order'), data.get('max_per_order'))
Item.clean_available(data.get('available_from'), data.get('available_until')) Item.clean_available(data.get('available_from'), data.get('available_until'))
if data.get('personalized') and not data.get('admission'):
raise ValidationError(_('Only admission products can currently be personalized.'))
if data.get('admission') and 'personalized' not in data and not self.instance:
# Backwards compatibility
data['personalized'] = True
elif 'admission' in data and not data['admission']:
data['personalized'] = False
if data.get('issue_giftcard'): if data.get('issue_giftcard'):
if data.get('tax_rule') and data.get('tax_rule').rate > 0: if data.get('tax_rule') and data.get('tax_rule').rate > 0:
raise ValidationError( raise ValidationError(
@@ -335,19 +261,9 @@ class ItemSerializer(I18nAwareModelSerializer):
for variation_data in variations_data: for variation_data in variations_data:
require_membership_types = variation_data.pop('require_membership_types', []) require_membership_types = variation_data.pop('require_membership_types', [])
var_meta_data = variation_data.pop('meta_data', {})
v = ItemVariation.objects.create(item=item, **variation_data) v = ItemVariation.objects.create(item=item, **variation_data)
if require_membership_types: if require_membership_types:
v.require_membership_types.add(*require_membership_types) v.require_membership_types.add(*require_membership_types)
if var_meta_data is not None:
for key, value in var_meta_data.items():
ItemVariationMetaValue.objects.create(
property=self.item_meta_properties.get(key),
value=value,
variation=v
)
for addon_data in addons_data: for addon_data in addons_data:
ItemAddOn.objects.create(base_item=item, **addon_data) ItemAddOn.objects.create(base_item=item, **addon_data)
for bundle_data in bundles_data: for bundle_data in bundles_data:

View File

@@ -29,7 +29,6 @@ import pycountry
from django.conf import settings from django.conf import settings
from django.core.files import File from django.core.files import File
from django.db.models import F, Q from django.db.models import F, Q
from django.utils.encoding import force_str
from django.utils.timezone import now from django.utils.timezone import now
from django.utils.translation import gettext_lazy from django.utils.translation import gettext_lazy
from django_countries.fields import Country from django_countries.fields import Country
@@ -62,25 +61,14 @@ from pretix.base.services.pricing import (
) )
from pretix.base.settings import COUNTRIES_WITH_STATE_IN_ADDRESS from pretix.base.settings import COUNTRIES_WITH_STATE_IN_ADDRESS
from pretix.base.signals import register_ticket_outputs from pretix.base.signals import register_ticket_outputs
from pretix.helpers.countries import CachedCountries
from pretix.multidomain.urlreverse import build_absolute_uri from pretix.multidomain.urlreverse import build_absolute_uri
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
class CompatibleCountryField(serializers.Field): class CompatibleCountryField(serializers.Field):
countries = CachedCountries()
default_error_messages = {
'invalid_choice': gettext_lazy('"{input}" is not a valid choice.')
}
def to_internal_value(self, data): def to_internal_value(self, data):
country = self.countries.alpha2(data) return {self.field_name: Country(data)}
if data and not country:
country = self.countries.by_name(force_str(data))
if not country:
self.fail("invalid_choice", input=data)
return {self.field_name: Country(country)}
def to_representation(self, instance: InvoiceAddress): def to_representation(self, instance: InvoiceAddress):
if instance.country: if instance.country:
@@ -118,10 +106,6 @@ class InvoiceAddressSerializer(I18nAwareModelSerializer):
raise ValidationError( raise ValidationError(
{'name': ['Do not specify name if you specified name_parts.']} {'name': ['Do not specify name if you specified name_parts.']}
) )
if data.get('name_parts') and not isinstance(data.get('name_parts'), dict):
raise ValidationError({'name_parts': ['Invalid data type']})
if data.get('name_parts') and '_scheme' not in data.get('name_parts'): if data.get('name_parts') and '_scheme' not in data.get('name_parts'):
data['name_parts']['_scheme'] = self.context['request'].event.settings.name_scheme data['name_parts']['_scheme'] = self.context['request'].event.settings.name_scheme
@@ -375,19 +359,10 @@ class PdfDataSerializer(serializers.Field):
for k, v in ev._cached_meta_data.items(): for k, v in ev._cached_meta_data.items():
res['meta:' + k] = v res['meta:' + k] = v
if instance.variation_id: if not hasattr(instance.item, '_cached_meta_data'):
print(instance, instance.variation, instance.variation_id, instance.item) instance.item._cached_meta_data = instance.item.meta_data
if not hasattr(instance.variation, '_cached_meta_data'): for k, v in instance.item._cached_meta_data.items():
instance.variation.item = instance.item # saves some database lookups res['itemmeta:' + k] = v
instance.variation._cached_meta_data = instance.variation.meta_data
print(instance.variation._cached_meta_data.items())
for k, v in instance.variation._cached_meta_data.items():
res['itemmeta:' + k] = v
else:
if not hasattr(instance.item, '_cached_meta_data'):
instance.item._cached_meta_data = instance.item.meta_data
for k, v in instance.item._cached_meta_data.items():
res['itemmeta:' + k] = v
res['images'] = {} res['images'] = {}
@@ -435,13 +410,13 @@ class OrderPositionSerializer(I18nAwareModelSerializer):
class Meta: class Meta:
model = OrderPosition model = OrderPosition
fields = ('id', 'order', 'positionid', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts', fields = ('id', 'order', 'positionid', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts',
'company', 'street', 'zipcode', 'city', 'country', 'state', 'discount', 'company', 'street', 'zipcode', 'city', 'country', 'state',
'attendee_email', 'voucher', 'tax_rate', 'tax_value', 'secret', 'addon_to', 'subevent', 'checkins', 'attendee_email', 'voucher', 'tax_rate', 'tax_value', 'secret', 'addon_to', 'subevent', 'checkins',
'downloads', 'answers', 'tax_rule', 'pseudonymization_id', 'pdf_data', 'seat', 'canceled') 'downloads', 'answers', 'tax_rule', 'pseudonymization_id', 'pdf_data', 'seat', 'canceled')
read_only_fields = ( read_only_fields = (
'id', 'order', 'positionid', 'item', 'variation', 'price', 'voucher', 'tax_rate', 'tax_value', 'secret', 'id', 'order', 'positionid', 'item', 'variation', 'price', 'voucher', 'tax_rate', 'tax_value', 'secret',
'addon_to', 'subevent', 'checkins', 'downloads', 'answers', 'tax_rule', 'pseudonymization_id', 'pdf_data', 'addon_to', 'subevent', 'checkins', 'downloads', 'answers', 'tax_rule', 'pseudonymization_id', 'pdf_data',
'seat', 'canceled', 'discount', 'seat', 'canceled'
) )
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
@@ -578,22 +553,12 @@ class OrderPaymentSerializer(I18nAwareModelSerializer):
'details') 'details')
class RefundDetailsField(serializers.Field):
def to_representation(self, value: OrderRefund):
pp = value.payment_provider
if not pp:
return {}
return pp.api_refund_details(value)
class OrderRefundSerializer(I18nAwareModelSerializer): class OrderRefundSerializer(I18nAwareModelSerializer):
payment = SlugRelatedField(slug_field='local_id', read_only=True) payment = SlugRelatedField(slug_field='local_id', read_only=True)
details = RefundDetailsField(source='*', allow_null=True, read_only=True)
class Meta: class Meta:
model = OrderRefund model = OrderRefund
fields = ('local_id', 'state', 'source', 'amount', 'payment', 'created', 'execution_date', 'comment', 'provider', fields = ('local_id', 'state', 'source', 'amount', 'payment', 'created', 'execution_date', 'comment', 'provider')
'details')
class OrderURLField(serializers.URLField): class OrderURLField(serializers.URLField):
@@ -635,32 +600,6 @@ class OrderSerializer(I18nAwareModelSerializer):
if not self.context['pdf_data']: if not self.context['pdf_data']:
self.fields['positions'].child.fields.pop('pdf_data', None) self.fields['positions'].child.fields.pop('pdf_data', None)
includes = set(self.context['include'])
if includes:
for fname, field in list(self.fields.items()):
if fname in includes:
continue
elif hasattr(field, 'child'): # Nested list serializers
found_any = False
for childfname, childfield in list(field.child.fields.items()):
if f'{fname}.{childfname}' not in includes:
field.child.fields.pop(childfname)
else:
found_any = True
if not found_any:
self.fields.pop(fname)
elif isinstance(field, serializers.Serializer): # Nested serializers
found_any = False
for childfname, childfield in list(field.fields.items()):
if f'{fname}.{childfname}' not in includes:
field.fields.pop(childfname)
else:
found_any = True
if not found_any:
self.fields.pop(fname)
else:
self.fields.pop(fname)
for exclude_field in self.context['exclude']: for exclude_field in self.context['exclude']:
p = exclude_field.split('.') p = exclude_field.split('.')
if p[0] in self.fields: if p[0] in self.fields:
@@ -782,7 +721,7 @@ class OrderPositionCreateSerializer(I18nAwareModelSerializer):
class Meta: class Meta:
model = OrderPosition model = OrderPosition
fields = ('positionid', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts', 'attendee_email', fields = ('positionid', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts', 'attendee_email',
'company', 'street', 'zipcode', 'city', 'country', 'state', 'is_bundled', 'company', 'street', 'zipcode', 'city', 'country', 'state',
'secret', 'addon_to', 'subevent', 'answers', 'seat', 'voucher') 'secret', 'addon_to', 'subevent', 'answers', 'seat', 'voucher')
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
@@ -845,10 +784,6 @@ class OrderPositionCreateSerializer(I18nAwareModelSerializer):
raise ValidationError( raise ValidationError(
{'attendee_name': ['Do not specify attendee_name if you specified attendee_name_parts.']} {'attendee_name': ['Do not specify attendee_name if you specified attendee_name_parts.']}
) )
if data.get('attendee_name_parts') and not isinstance(data.get('attendee_name_parts'), dict):
raise ValidationError({'attendee_name_parts': ['Invalid data type']})
if data.get('attendee_name_parts') and '_scheme' not in data.get('attendee_name_parts'): if data.get('attendee_name_parts') and '_scheme' not in data.get('attendee_name_parts'):
data['attendee_name_parts']['_scheme'] = self.context['request'].event.settings.name_scheme data['attendee_name_parts']['_scheme'] = self.context['request'].event.settings.name_scheme
@@ -1443,7 +1378,7 @@ class OrderCreateSerializer(I18nAwareModelSerializer):
state=OrderPayment.PAYMENT_STATE_CREATED state=OrderPayment.PAYMENT_STATE_CREATED
) )
order.create_transactions(is_new=True, fees=fees, positions=pos_map.values()) order.create_transactions(is_new=True, fees=fees, positions=pos_map.values(), source=self.context['source'])
return order return order

View File

@@ -158,14 +158,12 @@ class OrderPositionInfoPatchSerializer(serializers.ModelSerializer):
a.question_id: a for a in instance.answers.all() a.question_id: a for a in instance.answers.all()
} }
for answ_data in answers_data: for answ_data in answers_data:
if not answ_data.get('answer'):
continue
options = answ_data.pop('options', []) options = answ_data.pop('options', [])
if answ_data['question'].pk in qs_seen: if answ_data['question'].pk in qs_seen:
raise ValidationError(f'Question {answ_data["question"]} was sent twice.') raise ValidationError(f'Question {answ_data["question"]} was sent twice.')
if answ_data['question'].pk in answercache: if answ_data['question'].pk in answercache:
a = answercache[answ_data['question'].pk] a = answercache[answ_data['question'].pk]
if isinstance(answ_data.get('answer'), File): if isinstance(answ_data['answer'], File):
a.file.save(answ_data['answer'].name, answ_data['answer'], save=False) a.file.save(answ_data['answer'].name, answ_data['answer'], save=False)
a.answer = 'file://' + a.file.name a.answer = 'file://' + a.file.name
elif a.answer.startswith('file://') and answ_data['answer'] == "file:keep": elif a.answer.startswith('file://') and answ_data['answer'] == "file:keep":
@@ -175,7 +173,7 @@ class OrderPositionInfoPatchSerializer(serializers.ModelSerializer):
setattr(a, attr, value) setattr(a, attr, value)
a.save() a.save()
else: else:
if isinstance(answ_data.get('answer'), File): if isinstance(answ_data['answer'], File):
an = answ_data.pop('answer') an = answ_data.pop('answer')
a = instance.answers.create(**answ_data, answer='') a = instance.answers.create(**answ_data, answer='')
a.file.save(os.path.basename(an.name), an, save=False) a.file.save(os.path.basename(an.name), an, save=False)

View File

@@ -79,13 +79,6 @@ class CustomerSerializer(I18nAwareModelSerializer):
validated_data['external_identifier'] = instance.external_identifier validated_data['external_identifier'] = instance.external_identifier
return super().update(instance, validated_data) return super().update(instance, validated_data)
def validate(self, data):
if data.get('name_parts') and not isinstance(data.get('name_parts'), dict):
raise ValidationError({'name_parts': ['Invalid data type']})
if data.get('name_parts') and '_scheme' not in data.get('name_parts'):
data['name_parts']['_scheme'] = self.context['request'].organizer.settings.name_scheme
return data
class CustomerCreateSerializer(CustomerSerializer): class CustomerCreateSerializer(CustomerSerializer):
send_email = serializers.BooleanField(default=False, required=False, allow_null=True) send_email = serializers.BooleanField(default=False, required=False, allow_null=True)

View File

@@ -61,7 +61,7 @@ class VoucherSerializer(I18nAwareModelSerializer):
class Meta: class Meta:
model = Voucher model = Voucher
fields = ('id', 'code', 'max_usages', 'redeemed', 'min_usages', 'valid_until', 'block_quota', fields = ('id', 'code', 'max_usages', 'redeemed', 'valid_until', 'block_quota',
'allow_ignore_quota', 'price_mode', 'value', 'item', 'variation', 'quota', 'allow_ignore_quota', 'price_mode', 'value', 'item', 'variation', 'quota',
'tag', 'comment', 'subevent', 'show_hidden_items', 'seat') 'tag', 'comment', 'subevent', 'show_hidden_items', 'seat')
read_only_fields = ('id', 'redeemed') read_only_fields = ('id', 'redeemed')

View File

@@ -35,8 +35,7 @@
import importlib import importlib
from django.apps import apps from django.apps import apps
from django.conf.urls import re_path from django.conf.urls import include, re_path
from django.urls import include
from rest_framework import routers from rest_framework import routers
from pretix.api.views import cart from pretix.api.views import cart

14
src/pretix/api/utils.py Normal file
View File

@@ -0,0 +1,14 @@
from pretix.api.models import OAuthAccessToken
from pretix.base.models import Device, TeamAPIToken
def get_api_source(request):
if isinstance(request.auth, Device):
return "pretix.api", f"device:{request.auth.pk}"
elif isinstance(request.auth, TeamAPIToken):
return "pretix.api", f"token:{request.auth.pk}"
elif isinstance(request.auth, OAuthAccessToken):
return "pretix.api", f"oauth.app:{request.auth.application.pk}"
elif request.user.is_authenticated:
return "pretix.api", f"user:{request.user.pk}"
return "pretix.api", None

View File

@@ -92,11 +92,6 @@ class CartPositionViewSet(CreateModelMixin, DestroyModelMixin, viewsets.ReadOnly
def perform_create(self, serializer): def perform_create(self, serializer):
raise NotImplementedError() raise NotImplementedError()
@transaction.atomic()
def perform_destroy(self, instance):
instance.addons.all().delete()
instance.delete()
def _require_locking(self, quota_diff, voucher_use_diff, seat_diff): def _require_locking(self, quota_diff, voucher_use_diff, seat_diff):
if voucher_use_diff or seat_diff: if voucher_use_diff or seat_diff:
# If any vouchers or seats are used, we lock to make sure we don't redeem them to often # If any vouchers or seats are used, we lock to make sure we don't redeem them to often

View File

@@ -93,10 +93,8 @@ with scopes_disabled():
class CheckinListViewSet(viewsets.ModelViewSet): class CheckinListViewSet(viewsets.ModelViewSet):
serializer_class = CheckinListSerializer serializer_class = CheckinListSerializer
queryset = CheckinList.objects.none() queryset = CheckinList.objects.none()
filter_backends = (DjangoFilterBackend, RichOrderingFilter) filter_backends = (DjangoFilterBackend,)
filterset_class = CheckinListFilter filterset_class = CheckinListFilter
ordering = ('subevent__date_from', 'name', 'id')
ordering_fields = ('subevent__date_from', 'id', 'name',)
def _get_permission_name(self, request): def _get_permission_name(self, request):
if request.path.endswith('/failed_checkins/'): if request.path.endswith('/failed_checkins/'):
@@ -684,7 +682,7 @@ class CheckinListPositionViewSet(viewsets.ReadOnlyModelViewSet):
serializer_class = CheckinListOrderPositionSerializer serializer_class = CheckinListOrderPositionSerializer
queryset = OrderPosition.all.none() queryset = OrderPosition.all.none()
filter_backends = (ExtendedBackend, RichOrderingFilter) filter_backends = (ExtendedBackend, RichOrderingFilter)
ordering = (F('attendee_name_cached').asc(nulls_last=True), 'pk') ordering = (F('attendee_name_cached').asc(nulls_last=True), 'positionid')
ordering_fields = ( ordering_fields = (
'order__code', 'order__datetime', 'positionid', 'attendee_name', 'order__code', 'order__datetime', 'positionid', 'attendee_name',
'last_checked_in', 'order__email', 'last_checked_in', 'order__email',

View File

@@ -33,7 +33,6 @@
# License for the specific language governing permissions and limitations under the License. # License for the specific language governing permissions and limitations under the License.
import django_filters import django_filters
from django.conf import settings
from django.db import transaction from django.db import transaction
from django.db.models import Prefetch, ProtectedError, Q from django.db.models import Prefetch, ProtectedError, Q
from django.utils.timezone import now from django.utils.timezone import now
@@ -242,17 +241,13 @@ class EventViewSet(viewsets.ModelViewSet):
except Event.DoesNotExist: except Event.DoesNotExist:
raise ValidationError('Event to copy from was not found') raise ValidationError('Event to copy from was not found')
# Ensure that .installed() is only called when we NOT clone
plugins = serializer.validated_data.pop('plugins', None)
serializer.validated_data['plugins'] = None
new_event = serializer.save(organizer=self.request.organizer) new_event = serializer.save(organizer=self.request.organizer)
if copy_from: if copy_from:
new_event.copy_data_from(copy_from) new_event.copy_data_from(copy_from)
if plugins is not None: if 'plugins' in serializer.validated_data:
new_event.set_active_plugins(plugins) new_event.set_active_plugins(serializer.validated_data['plugins'])
if 'is_public' in serializer.validated_data: if 'is_public' in serializer.validated_data:
new_event.is_public = serializer.validated_data['is_public'] new_event.is_public = serializer.validated_data['is_public']
if 'testmode' in serializer.validated_data: if 'testmode' in serializer.validated_data:
@@ -261,17 +256,12 @@ class EventViewSet(viewsets.ModelViewSet):
new_event.sales_channels = serializer.validated_data['sales_channels'] new_event.sales_channels = serializer.validated_data['sales_channels']
if 'has_subevents' in serializer.validated_data: if 'has_subevents' in serializer.validated_data:
new_event.has_subevents = serializer.validated_data['has_subevents'] new_event.has_subevents = serializer.validated_data['has_subevents']
if 'date_admission' in serializer.validated_data:
new_event.date_admission = serializer.validated_data['date_admission']
new_event.save() new_event.save()
if 'timezone' in serializer.validated_data: if 'timezone' in serializer.validated_data:
new_event.settings.timezone = serializer.validated_data['timezone'] new_event.settings.timezone = serializer.validated_data['timezone']
else: else:
serializer.instance.set_defaults() serializer.instance.set_defaults()
new_event.set_active_plugins(plugins if plugins is not None else settings.PRETIX_PLUGINS_DEFAULT.split(','))
new_event.save(update_fields=['plugins'])
serializer.instance.log_action( serializer.instance.log_action(
'pretix.event.added', 'pretix.event.added',
user=self.request.user, user=self.request.user,
@@ -332,7 +322,6 @@ with scopes_disabled():
ends_after = django_filters.rest_framework.IsoDateTimeFilter(method='ends_after_qs') ends_after = django_filters.rest_framework.IsoDateTimeFilter(method='ends_after_qs')
modified_since = django_filters.IsoDateTimeFilter(field_name='last_modified', lookup_expr='gte') modified_since = django_filters.IsoDateTimeFilter(field_name='last_modified', lookup_expr='gte')
sales_channel = django_filters.rest_framework.CharFilter(method='sales_channel_qs') sales_channel = django_filters.rest_framework.CharFilter(method='sales_channel_qs')
search = django_filters.rest_framework.CharFilter(method='search_qs')
class Meta: class Meta:
model = SubEvent model = SubEvent
@@ -368,12 +357,6 @@ with scopes_disabled():
def sales_channel_qs(self, queryset, name, value): def sales_channel_qs(self, queryset, name, value):
return queryset.filter(event__sales_channels__contains=value) return queryset.filter(event__sales_channels__contains=value)
def search_qs(self, queryset, name, value):
return queryset.filter(
Q(name__icontains=i18ncomp(value))
| Q(location__icontains=i18ncomp(value))
)
class SubEventViewSet(ConditionalListView, viewsets.ModelViewSet): class SubEventViewSet(ConditionalListView, viewsets.ModelViewSet):
serializer_class = SubEventSerializer serializer_class = SubEventSerializer

View File

@@ -84,9 +84,7 @@ class ItemViewSet(ConditionalListView, viewsets.ModelViewSet):
def get_queryset(self): def get_queryset(self):
return self.request.event.items.select_related('tax_rule').prefetch_related( return self.request.event.items.select_related('tax_rule').prefetch_related(
'variations', 'addons', 'bundles', 'meta_values', 'meta_values__property', 'variations', 'addons', 'bundles', 'meta_values'
'variations__meta_values', 'variations__meta_values__property',
'require_membership_types', 'variations__require_membership_types',
).all() ).all()
def perform_create(self, serializer): def perform_create(self, serializer):
@@ -149,11 +147,7 @@ class ItemVariationViewSet(viewsets.ModelViewSet):
return get_object_or_404(Item, pk=self.kwargs['item'], event=self.request.event) return get_object_or_404(Item, pk=self.kwargs['item'], event=self.request.event)
def get_queryset(self): def get_queryset(self):
return self.item.variations.all().prefetch_related( return self.item.variations.all()
'meta_values',
'meta_values__property',
'require_membership_types'
)
def get_serializer_context(self): def get_serializer_context(self):
ctx = super().get_serializer_context() ctx = super().get_serializer_context()

View File

@@ -34,7 +34,6 @@ from oauth2_provider.views import (
from pretix.api.models import OAuthApplication from pretix.api.models import OAuthApplication
from pretix.base.models import Organizer from pretix.base.models import Organizer
from pretix.control.views.user import RecentAuthenticationRequiredMixin
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@@ -55,7 +54,7 @@ class OAuthAllowForm(AllowForm):
del self.fields['organizers'] del self.fields['organizers']
class AuthorizationView(RecentAuthenticationRequiredMixin, BaseAuthorizationView): class AuthorizationView(BaseAuthorizationView):
template_name = "pretixcontrol/auth/oauth_authorization.html" template_name = "pretixcontrol/auth/oauth_authorization.html"
form_class = OAuthAllowForm form_class = OAuthAllowForm

View File

@@ -61,14 +61,13 @@ from pretix.api.serializers.orderchange import (
OrderPositionCreateForExistingOrderSerializer, OrderPositionCreateForExistingOrderSerializer,
OrderPositionInfoPatchSerializer, OrderPositionInfoPatchSerializer,
) )
from pretix.api.views import RichOrderingFilter from pretix.api.utils import get_api_source
from pretix.base.i18n import language from pretix.base.i18n import language
from pretix.base.models import ( from pretix.base.models import (
CachedCombinedTicket, CachedTicket, Checkin, Device, EventMetaValue, CachedCombinedTicket, CachedTicket, Checkin, Device, EventMetaValue,
Invoice, InvoiceAddress, ItemMetaValue, ItemVariation, Invoice, InvoiceAddress, ItemMetaValue, Order, OrderFee, OrderPayment,
ItemVariationMetaValue, Order, OrderFee, OrderPayment, OrderPosition, OrderPosition, OrderRefund, Quota, SubEvent, SubEventMetaValue, TaxRule,
OrderRefund, Quota, SubEvent, SubEventMetaValue, TaxRule, TeamAPIToken, TeamAPIToken, generate_secret,
generate_secret,
) )
from pretix.base.models.orders import QuestionAnswer, RevokedTicketSecret from pretix.base.models.orders import QuestionAnswer, RevokedTicketSecret
from pretix.base.payment import PaymentException from pretix.base.payment import PaymentException
@@ -192,7 +191,7 @@ class OrderViewSet(viewsets.ModelViewSet):
ctx['event'] = self.request.event ctx['event'] = self.request.event
ctx['pdf_data'] = self.request.query_params.get('pdf_data', 'false') == 'true' ctx['pdf_data'] = self.request.query_params.get('pdf_data', 'false') == 'true'
ctx['exclude'] = self.request.query_params.getlist('exclude') ctx['exclude'] = self.request.query_params.getlist('exclude')
ctx['include'] = self.request.query_params.getlist('include') ctx['source'] = get_api_source(self.request)
return ctx return ctx
def get_queryset(self): def get_queryset(self):
@@ -233,9 +232,7 @@ class OrderViewSet(viewsets.ModelViewSet):
Prefetch('item', queryset=self.request.event.items.prefetch_related( Prefetch('item', queryset=self.request.event.items.prefetch_related(
Prefetch('meta_values', ItemMetaValue.objects.select_related('property'), to_attr='meta_values_cached') Prefetch('meta_values', ItemMetaValue.objects.select_related('property'), to_attr='meta_values_cached')
)), )),
Prefetch('variation', queryset=ItemVariation.objects.prefetch_related( 'variation',
Prefetch('meta_values', ItemVariationMetaValue.objects.select_related('property'), to_attr='meta_values_cached')
)),
'answers', 'answers__options', 'answers__question', 'answers', 'answers__options', 'answers__question',
'item__category', 'item__category',
'addon_to__answers', 'addon_to__answers__options', 'addon_to__answers__question', 'addon_to__answers', 'addon_to__answers__options', 'addon_to__answers__question',
@@ -395,7 +392,8 @@ class OrderViewSet(viewsets.ModelViewSet):
oauth_application=request.auth.application if isinstance(request.auth, OAuthAccessToken) else None, oauth_application=request.auth.application if isinstance(request.auth, OAuthAccessToken) else None,
send_mail=send_mail, send_mail=send_mail,
email_comment=comment, email_comment=comment,
cancellation_fee=cancellation_fee cancellation_fee=cancellation_fee,
source=get_api_source(request),
) )
except OrderError as e: except OrderError as e:
return Response( return Response(
@@ -419,6 +417,7 @@ class OrderViewSet(viewsets.ModelViewSet):
order, order,
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None, auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None,
source=get_api_source(request),
) )
except OrderError as e: except OrderError as e:
return Response( return Response(
@@ -438,6 +437,7 @@ class OrderViewSet(viewsets.ModelViewSet):
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None, auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None,
send_mail=send_mail, send_mail=send_mail,
source=get_api_source(request),
) )
except Quota.QuotaExceededException as e: except Quota.QuotaExceededException as e:
return Response({'detail': str(e)}, status=status.HTTP_400_BAD_REQUEST) return Response({'detail': str(e)}, status=status.HTTP_400_BAD_REQUEST)
@@ -458,6 +458,7 @@ class OrderViewSet(viewsets.ModelViewSet):
auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None, auth=request.auth if isinstance(request.auth, (Device, TeamAPIToken, OAuthAccessToken)) else None,
send_mail=send_mail, send_mail=send_mail,
comment=comment, comment=comment,
source=get_api_source(request),
) )
except OrderError as e: except OrderError as e:
return Response({'detail': str(e)}, status=status.HTTP_400_BAD_REQUEST) return Response({'detail': str(e)}, status=status.HTTP_400_BAD_REQUEST)
@@ -496,6 +497,7 @@ class OrderViewSet(viewsets.ModelViewSet):
order, order,
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=request.auth, auth=request.auth,
source=get_api_source(request),
) )
return self.retrieve(request, [], **kwargs) return self.retrieve(request, [], **kwargs)
@@ -513,6 +515,7 @@ class OrderViewSet(viewsets.ModelViewSet):
order, order,
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=(request.auth if isinstance(request.auth, (TeamAPIToken, OAuthAccessToken, Device)) else None), auth=(request.auth if isinstance(request.auth, (TeamAPIToken, OAuthAccessToken, Device)) else None),
source=get_api_source(request),
) )
return self.retrieve(request, [], **kwargs) return self.retrieve(request, [], **kwargs)
@@ -684,33 +687,28 @@ class OrderViewSet(viewsets.ModelViewSet):
) )
if order.require_approval: if order.require_approval:
email_template = request.event.settings.mail_text_order_placed_require_approval email_template = request.event.settings.mail_text_order_placed_require_approval
subject_template = request.event.settings.mail_subject_order_placed_require_approval
log_entry = 'pretix.event.order.email.order_placed_require_approval' log_entry = 'pretix.event.order.email.order_placed_require_approval'
email_attendees = False email_attendees = False
elif free_flow: elif free_flow:
email_template = request.event.settings.mail_text_order_free email_template = request.event.settings.mail_text_order_free
subject_template = request.event.settings.mail_subject_order_free
log_entry = 'pretix.event.order.email.order_free' log_entry = 'pretix.event.order.email.order_free'
email_attendees = request.event.settings.mail_send_order_free_attendee email_attendees = request.event.settings.mail_send_order_free_attendee
email_attendees_template = request.event.settings.mail_text_order_free_attendee email_attendees_template = request.event.settings.mail_text_order_free_attendee
subject_attendees_template = request.event.settings.mail_subject_order_free_attendee
else: else:
email_template = request.event.settings.mail_text_order_placed email_template = request.event.settings.mail_text_order_placed
subject_template = request.event.settings.mail_subject_order_placed
log_entry = 'pretix.event.order.email.order_placed' log_entry = 'pretix.event.order.email.order_placed'
email_attendees = request.event.settings.mail_send_order_placed_attendee email_attendees = request.event.settings.mail_send_order_placed_attendee
email_attendees_template = request.event.settings.mail_text_order_placed_attendee email_attendees_template = request.event.settings.mail_text_order_placed_attendee
subject_attendees_template = request.event.settings.mail_subject_order_placed_attendee
_order_placed_email( _order_placed_email(
request.event, order, email_template, subject_template, request.event, order, payment.payment_provider if payment else None, email_template,
log_entry, invoice, [payment] if payment else [], is_free=free_flow log_entry, invoice, payment, is_free=free_flow
) )
if email_attendees: if email_attendees:
for p in order.positions.all(): for p in order.positions.all():
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email: if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
_order_placed_email_attendee(request.event, order, p, email_attendees_template, subject_attendees_template, _order_placed_email_attendee(request.event, order, p, email_attendees_template, log_entry,
log_entry, is_free=free_flow) is_free=free_flow)
if not free_flow and order.status == Order.STATUS_PAID and payment: if not free_flow and order.status == Order.STATUS_PAID and payment:
payment._send_paid_mail(invoice, None, '') payment._send_paid_mail(invoice, None, '')
@@ -940,7 +938,7 @@ with scopes_disabled():
class OrderPositionViewSet(viewsets.ModelViewSet): class OrderPositionViewSet(viewsets.ModelViewSet):
serializer_class = OrderPositionSerializer serializer_class = OrderPositionSerializer
queryset = OrderPosition.all.none() queryset = OrderPosition.all.none()
filter_backends = (DjangoFilterBackend, RichOrderingFilter) filter_backends = (DjangoFilterBackend, OrderingFilter)
ordering = ('order__datetime', 'positionid') ordering = ('order__datetime', 'positionid')
ordering_fields = ('order__code', 'order__datetime', 'positionid', 'attendee_name', 'order__status',) ordering_fields = ('order__code', 'order__datetime', 'positionid', 'attendee_name', 'order__status',)
filterset_class = OrderPositionFilter filterset_class = OrderPositionFilter
@@ -1002,11 +1000,7 @@ class OrderPositionViewSet(viewsets.ModelViewSet):
Prefetch('meta_values', ItemMetaValue.objects.select_related('property'), Prefetch('meta_values', ItemMetaValue.objects.select_related('property'),
to_attr='meta_values_cached') to_attr='meta_values_cached')
)), )),
Prefetch('variation', queryset=self.request.event.items.prefetch_related( 'variation', 'answers', 'answers__options', 'answers__question',
Prefetch('meta_values', ItemVariationMetaValue.objects.select_related('property'),
to_attr='meta_values_cached')
)),
'answers', 'answers__options', 'answers__question',
'item__category', 'item__category',
Prefetch('subevent', queryset=self.request.event.subevents.prefetch_related( Prefetch('subevent', queryset=self.request.event.subevents.prefetch_related(
Prefetch('meta_values', to_attr='meta_values_cached', Prefetch('meta_values', to_attr='meta_values_cached',
@@ -1498,7 +1492,8 @@ class PaymentViewSet(CreateModelMixin, viewsets.ReadOnlyModelViewSet):
if mark_refunded: if mark_refunded:
mark_order_refunded(payment.order, mark_order_refunded(payment.order,
user=self.request.user if self.request.user.is_authenticated else None, user=self.request.user if self.request.user.is_authenticated else None,
auth=self.request.auth) auth=self.request.auth,
source=get_api_source(self.request))
else: else:
payment.order.status = Order.STATUS_PENDING payment.order.status = Order.STATUS_PENDING
payment.order.set_expires( payment.order.set_expires(
@@ -1571,7 +1566,7 @@ class RefundViewSet(CreateModelMixin, viewsets.ReadOnlyModelViewSet):
mark_refunded = request.data.get('mark_canceled', False) mark_refunded = request.data.get('mark_canceled', False)
if mark_refunded: if mark_refunded:
mark_order_refunded(refund.order, user=self.request.user if self.request.user.is_authenticated else None, mark_order_refunded(refund.order, user=self.request.user if self.request.user.is_authenticated else None,
auth=self.request.auth) auth=self.request.auth, source=get_api_source(self.request))
elif not (refund.order.status == Order.STATUS_PAID and refund.order.pending_sum <= 0): elif not (refund.order.status == Order.STATUS_PAID and refund.order.pending_sum <= 0):
refund.order.status = Order.STATUS_PENDING refund.order.status = Order.STATUS_PENDING
refund.order.set_expires( refund.order.set_expires(
@@ -1618,23 +1613,13 @@ class RefundViewSet(CreateModelMixin, viewsets.ReadOnlyModelViewSet):
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=request.auth auth=request.auth
) )
if r.state in (OrderRefund.REFUND_STATE_DONE, OrderRefund.REFUND_STATE_CANCELED, OrderRefund.REFUND_STATE_FAILED):
r.order.log_action(
f'pretix.event.order.refund.{r.state}', {
'local_id': r.local_id,
'provider': r.provider,
},
user=request.user if request.user.is_authenticated else None,
auth=request.auth
)
if mark_refunded: if mark_refunded:
try: try:
mark_order_refunded( mark_order_refunded(
r.order, r.order,
user=request.user if request.user.is_authenticated else None, user=request.user if request.user.is_authenticated else None,
auth=(request.auth if request.auth else None), auth=(request.auth if request.auth else None),
source=get_api_source(self.request),
) )
except OrderError as e: except OrderError as e:
raise ValidationError(str(e)) raise ValidationError(str(e))

View File

@@ -51,7 +51,6 @@ from pretix.base.models import (
User, User,
) )
from pretix.base.settings import SETTINGS_AFFECTING_CSS from pretix.base.settings import SETTINGS_AFFECTING_CSS
from pretix.helpers import OF_SELF
from pretix.helpers.dicts import merge_dicts from pretix.helpers.dicts import merge_dicts
from pretix.presale.style import regenerate_organizer_css from pretix.presale.style import regenerate_organizer_css
@@ -179,7 +178,7 @@ class GiftCardViewSet(viewsets.ModelViewSet):
def perform_update(self, serializer): def perform_update(self, serializer):
if 'include_accepted' in self.request.GET: if 'include_accepted' in self.request.GET:
raise PermissionDenied("Accepted gift cards cannot be updated, use transact instead.") raise PermissionDenied("Accepted gift cards cannot be updated, use transact instead.")
GiftCard.objects.select_for_update(of=OF_SELF).get(pk=self.get_object().pk) GiftCard.objects.select_for_update().get(pk=self.get_object().pk)
old_value = serializer.instance.value old_value = serializer.instance.value
value = serializer.validated_data.pop('value') value = serializer.validated_data.pop('value')
inst = serializer.save(secret=serializer.instance.secret, currency=serializer.instance.currency, inst = serializer.save(secret=serializer.instance.secret, currency=serializer.instance.currency,
@@ -197,7 +196,7 @@ class GiftCardViewSet(viewsets.ModelViewSet):
@action(detail=True, methods=["POST"]) @action(detail=True, methods=["POST"])
@transaction.atomic() @transaction.atomic()
def transact(self, request, **kwargs): def transact(self, request, **kwargs):
gc = GiftCard.objects.select_for_update(of=OF_SELF).get(pk=self.get_object().pk) gc = GiftCard.objects.select_for_update().get(pk=self.get_object().pk)
value = serializers.DecimalField(max_digits=10, decimal_places=2).to_internal_value( value = serializers.DecimalField(max_digits=10, decimal_places=2).to_internal_value(
request.data.get('value') request.data.get('value')
) )

View File

@@ -21,7 +21,6 @@
# #
import datetime import datetime
from django.core.exceptions import ValidationError as DjangoValidationError
from django.utils.timezone import now from django.utils.timezone import now
from oauth2_provider.contrib.rest_framework import OAuth2Authentication from oauth2_provider.contrib.rest_framework import OAuth2Authentication
from rest_framework.authentication import SessionAuthentication from rest_framework.authentication import SessionAuthentication
@@ -34,9 +33,6 @@ from pretix.api.auth.device import DeviceTokenAuthentication
from pretix.api.auth.permission import AnyAuthenticatedClientPermission from pretix.api.auth.permission import AnyAuthenticatedClientPermission
from pretix.api.auth.token import TeamTokenAuthentication from pretix.api.auth.token import TeamTokenAuthentication
from pretix.base.models import CachedFile from pretix.base.models import CachedFile
from pretix.helpers.images import (
IMAGE_TYPES, validate_uploaded_file_for_valid_image,
)
ALLOWED_TYPES = { ALLOWED_TYPES = {
'image/gif': {'.gif'}, 'image/gif': {'.gif'},
@@ -65,13 +61,6 @@ class UploadView(APIView):
name=file_obj.name, name=file_obj.name,
type=content_type type=content_type
)) ))
if content_type in IMAGE_TYPES:
try:
validate_uploaded_file_for_valid_image(file_obj)
except DjangoValidationError as e:
raise ValidationError(e.message)
cf = CachedFile.objects.create( cf = CachedFile.objects.create(
expires=now() + datetime.timedelta(days=1), expires=now() + datetime.timedelta(days=1),
date=now(), date=now(),

View File

@@ -42,7 +42,6 @@ from pretix.base.models import LogEntry
from pretix.base.services.tasks import ProfiledTask, TransactionAwareTask from pretix.base.services.tasks import ProfiledTask, TransactionAwareTask
from pretix.base.signals import periodic_task from pretix.base.signals import periodic_task
from pretix.celery_app import app from pretix.celery_app import app
from pretix.helpers import OF_SELF
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
_ALL_EVENTS = None _ALL_EVENTS = None
@@ -503,8 +502,7 @@ def manually_retry_all_calls(webhook_id: int):
webhook = WebHook.objects.get(id=webhook_id) webhook = WebHook.objects.get(id=webhook_id)
with scope(organizer=webhook.organizer), transaction.atomic(): with scope(organizer=webhook.organizer), transaction.atomic():
for whcr in webhook.retries.select_for_update( for whcr in webhook.retries.select_for_update(
skip_locked=connection.features.has_select_for_update_skip_locked, skip_locked=connection.features.has_select_for_update_skip_locked
of=OF_SELF
): ):
send_webhook.apply_async( send_webhook.apply_async(
args=(whcr.logentry_id, whcr.action_type, whcr.webhook_id, whcr.retry_count), args=(whcr.logentry_id, whcr.action_type, whcr.webhook_id, whcr.retry_count),
@@ -517,8 +515,7 @@ def manually_retry_all_calls(webhook_id: int):
def schedule_webhook_retries_on_celery(sender, **kwargs): def schedule_webhook_retries_on_celery(sender, **kwargs):
with transaction.atomic(): with transaction.atomic():
for whcr in WebHookCallRetry.objects.select_for_update( for whcr in WebHookCallRetry.objects.select_for_update(
skip_locked=connection.features.has_select_for_update_skip_locked, skip_locked=connection.features.has_select_for_update_skip_locked
of=OF_SELF
).filter(retry_not_before__lt=now()): ).filter(retry_not_before__lt=now()):
send_webhook.apply_async( send_webhook.apply_async(
args=(whcr.logentry_id, whcr.action_type, whcr.webhook_id, whcr.retry_count), args=(whcr.logentry_id, whcr.action_type, whcr.webhook_id, whcr.retry_count),

View File

@@ -1,225 +0,0 @@
#
# This file is part of pretix (Community Edition).
#
# Copyright (C) 2014-2020 Raphael Michel and contributors
# Copyright (C) 2020-2021 rami.io GmbH and contributors
#
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
# Public License as published by the Free Software Foundation in version 3 of the License.
#
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
# this file, see <https://pretix.eu/about/en/license>.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
# <https://www.gnu.org/licenses/>.
#
from collections import defaultdict
from django.core.exceptions import ValidationError
from django.utils.translation import gettext_lazy as _
from localflavor.ar.forms import ARPostalCodeField
from localflavor.at.forms import ATZipCodeField
from localflavor.au.forms import AUPostCodeField
from localflavor.be.forms import BEPostalCodeField
from localflavor.br.forms import BRZipCodeField
from localflavor.ca.forms import CAPostalCodeField
from localflavor.ch.forms import CHZipCodeField
from localflavor.cn.forms import CNPostCodeField
from localflavor.cu.forms import CUPostalCodeField
from localflavor.cz.forms import CZPostalCodeField
from localflavor.de.forms import DEZipCodeField
from localflavor.dk.forms import DKPostalCodeField
from localflavor.ee.forms import EEZipCodeField
from localflavor.es.forms import ESPostalCodeField
from localflavor.fi.forms import FIZipCodeField
from localflavor.fr.forms import FRZipCodeField
from localflavor.gb.forms import GBPostcodeField
from localflavor.gr.forms import GRPostalCodeField
from localflavor.hr.forms import HRPostalCodeField
from localflavor.ie.forms import EircodeField
from localflavor.il.forms import ILPostalCodeField
from localflavor.in_.forms import INZipCodeField
from localflavor.ir.forms import IRPostalCodeField
from localflavor.is_.is_postalcodes import IS_POSTALCODES
from localflavor.it.forms import ITZipCodeField
from localflavor.jp.forms import JPPostalCodeField
from localflavor.lt.forms import LTPostalCodeField
from localflavor.lv.forms import LVPostalCodeField
from localflavor.ma.forms import MAPostalCodeField
from localflavor.mt.forms import MTPostalCodeField
from localflavor.mx.forms import MXZipCodeField
from localflavor.nl.forms import NLZipCodeField
from localflavor.no.forms import NOZipCodeField
from localflavor.nz.forms import NZPostCodeField
from localflavor.pk.forms import PKPostCodeField
from localflavor.pl.forms import PLPostalCodeField
from localflavor.pt.forms import PTZipCodeField
from localflavor.ro.forms import ROPostalCodeField
from localflavor.ru.forms import RUPostalCodeField
from localflavor.se.forms import SEPostalCodeField
from localflavor.sg.forms import SGPostCodeField
from localflavor.si.si_postalcodes import SI_POSTALCODES
from localflavor.sk.forms import SKPostalCodeField
from localflavor.tr.forms import TRPostalCodeField
from localflavor.ua.forms import UAPostalCodeField
from localflavor.us.forms import USZipCodeField
from localflavor.za.forms import ZAPostCodeField
from pretix.base.settings import COUNTRIES_WITH_STATE_IN_ADDRESS
_validator_classes = defaultdict(list)
COUNTRIES_WITH_STREET_ZIPCODE_AND_CITY_REQUIRED = {
# We don't presume this for countries we don't have knowledge about, there are countries in the
# world e.g. without zipcodes
'AR', 'AT', 'AU', 'BE', 'BR', 'CA', 'CH', 'CN', 'CU', 'CZ', 'DE', 'DK', 'EE', 'ES', 'FI', 'FR',
'GB', 'GR', 'HR', 'IE', 'IL', 'IN', 'IR', 'IS', 'IT', 'JP', 'LT', 'LV', 'MA', 'MT', 'MX', 'NL',
'NO', 'NZ', 'PK', 'PL', 'PT', 'RO', 'RU', 'SE', 'SG', 'SI', 'SK', 'TR', 'UA', 'US', 'ZA',
}
def validate_address(address: dict, all_optional=False):
"""
:param address: A dictionary with at least the entries ``street``, ``zipcode``, ``city``, ``country``,
``state``
:return: The dictionary, possibly with changes
"""
if not address.get('street') and not address.get('zipcode') and not address.get('city'):
# Consider the actual address part to be empty, no further validation necessary, if the
# address should be required, it's the callers job to validate that at least one of these
# fields is filled
return address
if not address.get('country'):
raise ValidationError({'country': [_('This field is required.')]})
if str(address['country']) in COUNTRIES_WITH_STATE_IN_ADDRESS and not address.get('state') and not all_optional:
raise ValidationError({'state': [_('This field is required.')]})
if str(address['country']) in COUNTRIES_WITH_STREET_ZIPCODE_AND_CITY_REQUIRED and not all_optional:
for f in ('street', 'zipcode', 'city'):
if not address.get(f):
raise ValidationError({f: [_('This field is required.')]})
for klass in _validator_classes[str(address['country'])]:
validator = klass()
try:
if address.get('zipcode'):
address['zipcode'] = validator.validate_zipcode(address['zipcode'])
except ValidationError as e:
raise ValidationError({'zipcode': list(e)})
return address
def register_validator_for(country):
def inner(klass):
_validator_classes[country].append(klass)
return klass
return inner
class BaseValidator:
required_fields = []
def validate_zipcode(self, value):
return value
"""
Currently, mostly have validators that are auto-generated from django-localflavor
but custom ones can be added like this:
@register_validator_for('DE')
class DEValidator(BaseValidator):
def validate_zipcode(value):
return value
In the future, we can also add additional methods to validate that e.g. a city
is plausible for a given zip code.
"""
_zip_code_fields = {
'AR': ARPostalCodeField,
'AT': ATZipCodeField,
'AU': AUPostCodeField,
'BE': BEPostalCodeField,
'BR': BRZipCodeField,
'CA': CAPostalCodeField,
'CH': CHZipCodeField,
'CN': CNPostCodeField,
'CU': CUPostalCodeField,
'CZ': CZPostalCodeField,
'DE': DEZipCodeField,
'DK': DKPostalCodeField,
'EE': EEZipCodeField,
'ES': ESPostalCodeField,
'FI': FIZipCodeField,
'FR': FRZipCodeField,
'GB': GBPostcodeField,
'GR': GRPostalCodeField,
'HR': HRPostalCodeField,
'IE': EircodeField,
'IL': ILPostalCodeField,
'IN': INZipCodeField,
'IR': IRPostalCodeField,
'IT': ITZipCodeField,
'JP': JPPostalCodeField,
'LT': LTPostalCodeField,
'LV': LVPostalCodeField,
'MA': MAPostalCodeField,
'MT': MTPostalCodeField,
'MX': MXZipCodeField,
'NL': NLZipCodeField,
'NO': NOZipCodeField,
'NZ': NZPostCodeField,
'PK': PKPostCodeField,
'PL': PLPostalCodeField,
'PT': PTZipCodeField,
'RO': ROPostalCodeField,
'RU': RUPostalCodeField,
'SE': SEPostalCodeField,
'SG': SGPostCodeField,
'SK': SKPostalCodeField,
'TR': TRPostalCodeField,
'UA': UAPostalCodeField,
'US': USZipCodeField,
'ZA': ZAPostCodeField,
}
def _generate_class_from_zipcode_field(field_class):
class _GeneratedValidator(BaseValidator):
def validate_zipcode(self, value):
return field_class().clean(value)
return _GeneratedValidator
for cc, field_class in _zip_code_fields.items():
register_validator_for(cc)(_generate_class_from_zipcode_field(field_class))
@register_validator_for('IS')
class ISValidator(BaseValidator):
def validate_zipcode(self, value):
if value not in [entry[0] for entry in IS_POSTALCODES]:
raise ValidationError(_('Enter a postal code in the format XXX.'), code='invalid')
return value
@register_validator_for('SI')
class SIValidator(BaseValidator):
def validate_zipcode(self, value):
try:
if int(value) not in [entry[0] for entry in SI_POSTALCODES]:
raise ValidationError(_('Enter a postal code in the format XXXX.'), code='invalid')
except ValueError:
raise ValidationError(_('Enter a postal code in the format XXXX.'), code='invalid')
return value

View File

@@ -320,18 +320,13 @@ def get_email_context(**kwargs):
return ctx return ctx
def _placeholder_payments(order, payments): def _placeholder_payment(order, payment):
d = [] if not payment:
for payment in payments: return None
if 'payment' in inspect.signature(payment.payment_provider.order_pending_mail_render).parameters: if 'payment' in inspect.signature(payment.payment_provider.order_pending_mail_render).parameters:
d.append(str(payment.payment_provider.order_pending_mail_render(order, payment))) return str(payment.payment_provider.order_pending_mail_render(order, payment))
else:
d.append(str(payment.payment_provider.order_pending_mail_render(order)))
d = [line for line in d if line.strip()]
if d:
return '\n\n'.join(d)
else: else:
return '' return str(payment.payment_provider.order_pending_mail_render(order))
def get_best_name(position_or_address, parts=False): def get_best_name(position_or_address, parts=False):
@@ -381,14 +376,6 @@ def base_placeholders(sender, **kwargs):
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'currency', ['event'], lambda event: event.currency, lambda event: event.currency 'currency', ['event'], lambda event: event.currency, lambda event: event.currency
), ),
SimpleFunctionalMailTextPlaceholder(
'order_email', ['order'], lambda order: order.email, 'john@example.org'
),
SimpleFunctionalMailTextPlaceholder(
'invoice_number', ['invoice'],
lambda invoice: invoice.full_invoice_no,
f'{sender.settings.invoice_numbers_prefix or (sender.slug.upper() + "-")}00000'
),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'refund_amount', ['event_or_subevent', 'refund_amount'], 'refund_amount', ['event_or_subevent', 'refund_amount'],
lambda event_or_subevent, refund_amount: LazyCurrencyNumber(refund_amount, event_or_subevent.currency), lambda event_or_subevent, refund_amount: LazyCurrencyNumber(refund_amount, event_or_subevent.currency),
@@ -520,20 +507,20 @@ def base_placeholders(sender, **kwargs):
lambda event: (event if not event.has_subevents or not event.subevents.exists() else event.subevents.first()).get_date_from_display() lambda event: (event if not event.has_subevents or not event.subevents.exists() else event.subevents.first()).get_date_from_display()
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'url_remove', ['waiting_list_voucher', 'event'], 'url_remove', ['waiting_list_entry', 'event'],
lambda waiting_list_voucher, event: build_absolute_uri( lambda waiting_list_entry, event: build_absolute_uri(
event, 'presale:event.waitinglist.remove' event, 'presale:event.waitinglist.remove'
) + '?voucher=' + waiting_list_voucher.code, ) + '?voucher=' + waiting_list_entry.voucher.code,
lambda event: build_absolute_uri( lambda event: build_absolute_uri(
event, event,
'presale:event.waitinglist.remove', 'presale:event.waitinglist.remove',
) + '?voucher=68CYU2H6ZTP3WLK5', ) + '?voucher=68CYU2H6ZTP3WLK5',
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'url', ['waiting_list_voucher', 'event'], 'url', ['waiting_list_entry', 'event'],
lambda waiting_list_voucher, event: build_absolute_uri( lambda waiting_list_entry, event: build_absolute_uri(
event, 'presale:event.redeem' event, 'presale:event.redeem'
) + '?voucher=' + waiting_list_voucher.code, ) + '?voucher=' + waiting_list_entry.voucher.code,
lambda event: build_absolute_uri( lambda event: build_absolute_uri(
event, event,
'presale:event.redeem', 'presale:event.redeem',
@@ -588,7 +575,7 @@ def base_placeholders(sender, **kwargs):
_('Sample Admission Ticket') _('Sample Admission Ticket')
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'code', ['waiting_list_voucher'], lambda waiting_list_voucher: waiting_list_voucher.code, 'code', ['waiting_list_entry'], lambda waiting_list_entry: waiting_list_entry.voucher.code,
'68CYU2H6ZTP3WLK5' '68CYU2H6ZTP3WLK5'
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
@@ -630,7 +617,7 @@ def base_placeholders(sender, **kwargs):
_('An individual text with a reason can be inserted here.'), _('An individual text with a reason can be inserted here.'),
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(
'payment_info', ['order', 'payments'], _placeholder_payments, 'payment_info', ['order', 'payment'], _placeholder_payment,
_('The amount has been charged to your card.'), _('The amount has been charged to your card.'),
), ),
SimpleFunctionalMailTextPlaceholder( SimpleFunctionalMailTextPlaceholder(

View File

@@ -36,7 +36,7 @@ import io
import tempfile import tempfile
from collections import OrderedDict, namedtuple from collections import OrderedDict, namedtuple
from decimal import Decimal from decimal import Decimal
from typing import Optional, Tuple from typing import Tuple
import pytz import pytz
from defusedcsv import csv from defusedcsv import csv
@@ -84,27 +84,6 @@ class BaseExporter:
""" """
raise NotImplementedError() # NOQA raise NotImplementedError() # NOQA
@property
def description(self) -> str:
"""
A description for this exporter.
"""
return ""
@property
def category(self) -> Optional[str]:
"""
A category name for this exporter, or ``None``.
"""
return None
@property
def featured(self) -> bool:
"""
If ``True``, this exporter will be highlighted.
"""
return False
@property @property
def identifier(self) -> str: def identifier(self) -> str:
""" """

View File

@@ -39,7 +39,7 @@ from zipfile import ZipFile
from django import forms from django import forms
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _
from pretix.base.models import QuestionAnswer from pretix.base.models import QuestionAnswer
@@ -49,10 +49,7 @@ from ..signals import register_data_exporters
class AnswerFilesExporter(BaseExporter): class AnswerFilesExporter(BaseExporter):
identifier = 'answerfiles' identifier = 'answerfiles'
verbose_name = _('Question answer file uploads') verbose_name = _('Answers to file upload questions')
category = pgettext_lazy('export_category', 'Order data')
description = _('Download a ZIP file including all files that have been uploaded by your customers while creating '
'an order.')
@property @property
def export_form_fields(self): def export_form_fields(self):

View File

@@ -36,7 +36,7 @@ from collections import OrderedDict
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.timezone import get_current_timezone from django.utils.timezone import get_current_timezone
from django.utils.translation import gettext as _, gettext_lazy, pgettext_lazy from django.utils.translation import gettext as _, gettext_lazy
from pretix.base.settings import PERSON_NAME_SCHEMES from pretix.base.settings import PERSON_NAME_SCHEMES
@@ -48,8 +48,6 @@ class CustomerListExporter(OrganizerLevelExportMixin, ListExporter):
identifier = 'customerlist' identifier = 'customerlist'
verbose_name = gettext_lazy('Customer accounts') verbose_name = gettext_lazy('Customer accounts')
organizer_required_permission = 'can_manage_customers' organizer_required_permission = 'can_manage_customers'
category = pgettext_lazy('export_category', 'Customer accounts')
description = gettext_lazy('Download a spreadsheet of all currently registered customer accounts.')
@property @property
def additional_form_fields(self): def additional_form_fields(self):

View File

@@ -23,24 +23,22 @@ import json
from collections import OrderedDict from collections import OrderedDict
from decimal import Decimal from decimal import Decimal
import dateutil
from django import forms
from django.core.serializers.json import DjangoJSONEncoder from django.core.serializers.json import DjangoJSONEncoder
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.timezone import now from django.utils.translation import gettext, gettext_lazy
from django.utils.translation import gettext, gettext_lazy, pgettext_lazy
from pretix.base.i18n import language from pretix.base.i18n import language
from pretix.base.models import Invoice, OrderPayment from pretix.base.models import Invoice, OrderPayment
from ..exporter import BaseExporter from ..exporter import BaseExporter
from ..signals import register_data_exporters from ..signals import register_data_exporters
from ..timeframes import DateFrameField, resolve_timeframe_to_dates_inclusive
class DekodiNREIExporter(BaseExporter): class DekodiNREIExporter(BaseExporter):
identifier = 'dekodi_nrei' identifier = 'dekodi_nrei'
verbose_name = 'dekodi NREI (JSON)' verbose_name = 'dekodi NREI (JSON)'
category = pgettext_lazy('export_category', 'Invoices')
description = gettext_lazy("Download invoices in a format that can be used by the dekodi NREI conversion software.")
# Specification: http://manuals.dekodi.de/nexuspub/schnittstellenbuch/ # Specification: http://manuals.dekodi.de/nexuspub/schnittstellenbuch/
@@ -115,7 +113,7 @@ class DekodiNREIExporter(BaseExporter):
'PTNo14': p.info_data.get('reference') or '', 'PTNo14': p.info_data.get('reference') or '',
'PTNo15': p.full_id or '', 'PTNo15': p.full_id or '',
}) })
elif p.provider and p.provider.startswith('stripe'): elif p.provider.startswith('stripe'):
src = p.info_data.get("source", p.info_data) src = p.info_data.get("source", p.info_data)
payments.append({ payments.append({
'PTID': '81', 'PTID': '81',
@@ -194,12 +192,17 @@ class DekodiNREIExporter(BaseExporter):
def render(self, form_data): def render(self, form_data):
qs = self.event.invoices.select_related('order').prefetch_related('lines', 'lines__subevent') qs = self.event.invoices.select_related('order').prefetch_related('lines', 'lines__subevent')
if form_data.get('date_range'): if form_data.get('date_from'):
d_start, d_end = resolve_timeframe_to_dates_inclusive(now(), form_data['date_range'], self.timezone) date_value = form_data.get('date_from')
if d_start: if isinstance(date_value, str):
qs = qs.filter(date__gte=d_start) date_value = dateutil.parser.parse(date_value).date()
if d_end: qs = qs.filter(date__gte=date_value)
qs = qs.filter(date__lte=d_end)
if form_data.get('date_to'):
date_value = form_data.get('date_to')
if isinstance(date_value, str):
date_value = dateutil.parser.parse(date_value).date()
qs = qs.filter(date__lte=date_value)
jo = { jo = {
'Format': 'NREI', 'Format': 'NREI',
@@ -215,14 +218,22 @@ class DekodiNREIExporter(BaseExporter):
def export_form_fields(self): def export_form_fields(self):
return OrderedDict( return OrderedDict(
[ [
('date_range', ('date_from',
DateFrameField( forms.DateField(
label=gettext_lazy('Date range'), label=gettext_lazy('Start date'),
include_future_frames=False, widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False, required=False,
help_text=gettext_lazy('Only include invoices issued in this time frame. Note that the invoice date does ' help_text=gettext_lazy('Only include invoices issued on or after this date. Note that the invoice date does '
'not always correspond to the order or payment date.') 'not always correspond to the order or payment date.')
)), )),
('date_to',
forms.DateField(
label=gettext_lazy('End date'),
widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False,
help_text=gettext_lazy('Only include invoices issued on or before this date. Note that the invoice date '
'does not always correspond to the order or payment date.')
)),
] ]
) )

View File

@@ -35,7 +35,7 @@
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.formats import date_format from django.utils.formats import date_format
from django.utils.functional import cached_property from django.utils.functional import cached_property
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _
from ...control.forms.filter import get_all_payment_providers from ...control.forms.filter import get_all_payment_providers
from ..exporter import ListExporter from ..exporter import ListExporter
@@ -45,8 +45,6 @@ from ..signals import register_multievent_data_exporters
class EventDataExporter(ListExporter): class EventDataExporter(ListExporter):
identifier = 'eventdata' identifier = 'eventdata'
verbose_name = _('Event data') verbose_name = _('Event data')
category = pgettext_lazy('export_category', 'Event data')
description = _('Download a spreadsheet with information on all events in this organizer account.')
@cached_property @cached_property
def providers(self): def providers(self):

View File

@@ -38,15 +38,13 @@ from collections import OrderedDict
from decimal import Decimal from decimal import Decimal
from zipfile import ZipFile from zipfile import ZipFile
import dateutil.parser
from django import forms from django import forms
from django.db.models import CharField, Exists, F, OuterRef, Q, Subquery, Sum from django.db.models import CharField, Exists, F, OuterRef, Q, Subquery, Sum
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.formats import date_format from django.utils.formats import date_format
from django.utils.functional import cached_property from django.utils.functional import cached_property
from django.utils.timezone import now from django.utils.translation import gettext, gettext_lazy as _, pgettext
from django.utils.translation import (
gettext, gettext_lazy as _, pgettext, pgettext_lazy,
)
from pretix.base.models import Invoice, InvoiceLine, OrderPayment from pretix.base.models import Invoice, InvoiceLine, OrderPayment
@@ -59,24 +57,30 @@ from ..services.invoices import invoice_pdf_task
from ..signals import ( from ..signals import (
register_data_exporters, register_multievent_data_exporters, register_data_exporters, register_multievent_data_exporters,
) )
from ..timeframes import DateFrameField, resolve_timeframe_to_dates_inclusive
class InvoiceExporterMixin: class InvoiceExporterMixin:
category = pgettext_lazy('export_category', 'Invoices')
@property @property
def invoice_exporter_form_fields(self): def invoice_exporter_form_fields(self):
return OrderedDict( return OrderedDict(
[ [
('date_range', ('date_from',
DateFrameField( forms.DateField(
label=_('Date range'), label=_('Start date'),
include_future_frames=False, widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False, required=False,
help_text=_('Only include invoices issued in this time frame. Note that the invoice date does ' help_text=_('Only include invoices issued on or after this date. Note that the invoice date does '
'not always correspond to the order or payment date.') 'not always correspond to the order or payment date.')
)), )),
('date_to',
forms.DateField(
label=_('End date'),
widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False,
help_text=_('Only include invoices issued on or before this date. Note that the invoice date '
'does not always correspond to the order or payment date.')
)),
('payment_provider', ('payment_provider',
forms.ChoiceField( forms.ChoiceField(
label=_('Payment provider'), label=_('Payment provider'),
@@ -108,12 +112,16 @@ class InvoiceExporterMixin:
) )
) )
qs = qs.filter(has_payment_with_provider=1) qs = qs.filter(has_payment_with_provider=1)
if form_data.get('date_range'): if form_data.get('date_from'):
d_start, d_end = resolve_timeframe_to_dates_inclusive(now(), form_data['date_range'], self.timezone) date_value = form_data.get('date_from')
if d_start: if isinstance(date_value, str):
qs = qs.filter(date__gte=d_start) date_value = dateutil.parser.parse(date_value).date()
if d_end: qs = qs.filter(date__gte=date_value)
qs = qs.filter(date__lte=d_end) if form_data.get('date_to'):
date_value = form_data.get('date_to')
if isinstance(date_value, str):
date_value = dateutil.parser.parse(date_value).date()
qs = qs.filter(date__lte=date_value)
return qs return qs
@@ -121,7 +129,6 @@ class InvoiceExporterMixin:
class InvoiceExporter(InvoiceExporterMixin, BaseExporter): class InvoiceExporter(InvoiceExporterMixin, BaseExporter):
identifier = 'invoices' identifier = 'invoices'
verbose_name = _('All invoices') verbose_name = _('All invoices')
description = _('Download all invoices created by the system as a ZIP file of PDF files.')
def render(self, form_data: dict, output_file=None): def render(self, form_data: dict, output_file=None):
qs = self.invoices_queryset(form_data).filter(shredded=False) qs = self.invoices_queryset(form_data).filter(shredded=False)
@@ -173,10 +180,6 @@ class InvoiceExporter(InvoiceExporterMixin, BaseExporter):
class InvoiceDataExporter(InvoiceExporterMixin, MultiSheetListExporter): class InvoiceDataExporter(InvoiceExporterMixin, MultiSheetListExporter):
identifier = 'invoicedata' identifier = 'invoicedata'
verbose_name = _('Invoice data') verbose_name = _('Invoice data')
description = _('Download a spreadsheet with the data of all invoices created by the system. The spreadsheet '
'includes two sheets, one with a line for every invoice, and one with a line for every position of '
'every invoice.')
featured = True
@property @property
def additional_form_fields(self): def additional_form_fields(self):

View File

@@ -22,14 +22,14 @@
from django.db.models import Prefetch from django.db.models import Prefetch
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.formats import date_format from django.utils.formats import date_format
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _
from openpyxl.styles import Alignment from openpyxl.styles import Alignment
from openpyxl.utils import get_column_letter from openpyxl.utils import get_column_letter
from ...helpers.safe_openpyxl import SafeCell from ...helpers.safe_openpyxl import SafeCell
from ..channels import get_all_sales_channels from ..channels import get_all_sales_channels
from ..exporter import ListExporter from ..exporter import ListExporter
from ..models import ItemMetaValue, ItemVariation, ItemVariationMetaValue from ..models import ItemMetaValue
from ..signals import register_data_exporters from ..signals import register_data_exporters
@@ -48,8 +48,6 @@ def _min(a1, a2):
class ItemDataExporter(ListExporter): class ItemDataExporter(ListExporter):
identifier = 'itemdata' identifier = 'itemdata'
verbose_name = _('Product data') verbose_name = _('Product data')
category = pgettext_lazy('export_category', 'Product data')
description = _('Download a spreadsheet with details about all products and variations.')
def iterate_list(self, form_data): def iterate_list(self, form_data):
locales = self.event.settings.locales locales = self.event.settings.locales
@@ -75,7 +73,6 @@ class ItemDataExporter(ListExporter):
_("Free price input"), _("Free price input"),
_("Sales tax"), _("Sales tax"),
_("Is an admission ticket"), _("Is an admission ticket"),
_("Personalized ticket"),
_("Generate tickets"), _("Generate tickets"),
_("Waiting list"), _("Waiting list"),
_("Available from"), _("Available from"),
@@ -109,27 +106,18 @@ class ItemDataExporter(ListExporter):
yield row yield row
for i in self.event.items.prefetch_related( for i in self.event.items.prefetch_related(
'variations',
Prefetch( Prefetch(
'meta_values', 'meta_values',
ItemMetaValue.objects.select_related('property'), ItemMetaValue.objects.select_related('property'),
to_attr='meta_values_cached' to_attr='meta_values_cached'
), )
Prefetch(
'variations',
queryset=ItemVariation.objects.prefetch_related(
Prefetch(
'meta_values',
ItemVariationMetaValue.objects.select_related('property'),
to_attr='meta_values_cached'
),
),
),
).select_related('category', 'tax_rule'): ).select_related('category', 'tax_rule'):
m = i.meta_data
vars = list(i.variations.all()) vars = list(i.variations.all())
if vars: if vars:
for v in vars: for v in vars:
m = v.meta_data
row = [ row = [
i.pk, i.pk,
v.pk, v.pk,
@@ -147,7 +135,6 @@ class ItemDataExporter(ListExporter):
_("Yes") if i.free_price else "", _("Yes") if i.free_price else "",
str(i.tax_rule) if i.tax_rule else "", str(i.tax_rule) if i.tax_rule else "",
_("Yes") if i.admission else "", _("Yes") if i.admission else "",
_("Yes") if i.personalized else "",
_("Yes") if i.generate_tickets else (_("Default") if i.generate_tickets is None else ""), _("Yes") if i.generate_tickets else (_("Default") if i.generate_tickets is None else ""),
_("Yes") if i.allow_waitinglist else "", _("Yes") if i.allow_waitinglist else "",
date_format(_max(i.available_from, v.available_from).astimezone(self.timezone), date_format(_max(i.available_from, v.available_from).astimezone(self.timezone),
@@ -173,7 +160,6 @@ class ItemDataExporter(ListExporter):
yield row yield row
else: else:
m = i.meta_data
row = [ row = [
i.pk, i.pk,
"", "",
@@ -191,7 +177,6 @@ class ItemDataExporter(ListExporter):
_("Yes") if i.free_price else "", _("Yes") if i.free_price else "",
str(i.tax_rule) if i.tax_rule else "", str(i.tax_rule) if i.tax_rule else "",
_("Yes") if i.admission else "", _("Yes") if i.admission else "",
_("Yes") if i.personalized else "",
_("Yes") if i.generate_tickets else (_("Default") if i.generate_tickets is None else ""), _("Yes") if i.generate_tickets else (_("Default") if i.generate_tickets is None else ""),
_("Yes") if i.allow_waitinglist else "", _("Yes") if i.allow_waitinglist else "",
date_format(i.available_from.astimezone(self.timezone), date_format(i.available_from.astimezone(self.timezone),

View File

@@ -36,22 +36,15 @@ import json
from decimal import Decimal from decimal import Decimal
from django.core.serializers.json import DjangoJSONEncoder from django.core.serializers.json import DjangoJSONEncoder
from django.db.models import Prefetch
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.functional import lazy
from django.utils.translation import gettext, gettext_lazy, pgettext_lazy
from ..exporter import BaseExporter from ..exporter import BaseExporter
from ..models import ItemMetaValue, ItemVariation, ItemVariationMetaValue
from ..signals import register_data_exporters from ..signals import register_data_exporters
class JSONExporter(BaseExporter): class JSONExporter(BaseExporter):
identifier = 'json' identifier = 'json'
verbose_name = lazy(lambda *args: gettext('Order data') + ' (JSON)', str)() verbose_name = 'Order data (JSON)'
category = pgettext_lazy('export_category', 'Order data')
description = gettext_lazy('Download a structured JSON representation of all orders. This might be useful for the '
'import in third-party systems.')
def render(self, form_data): def render(self, form_data):
jo = { jo = {
@@ -83,7 +76,6 @@ class JSONExporter(BaseExporter):
'tax_rate': item.tax_rule.rate if item.tax_rule else Decimal('0.00'), 'tax_rate': item.tax_rule.rate if item.tax_rule else Decimal('0.00'),
'tax_name': str(item.tax_rule.name) if item.tax_rule else None, 'tax_name': str(item.tax_rule.name) if item.tax_rule else None,
'admission': item.admission, 'admission': item.admission,
'personalized': item.personalized,
'active': item.active, 'active': item.active,
'sales_channels': item.sales_channels, 'sales_channels': item.sales_channels,
'description': str(item.description), 'description': str(item.description),
@@ -114,26 +106,9 @@ class JSONExporter(BaseExporter):
'available_from': variation.available_from, 'available_from': variation.available_from,
'available_until': variation.available_until, 'available_until': variation.available_until,
'hide_without_voucher': variation.hide_without_voucher, 'hide_without_voucher': variation.hide_without_voucher,
'meta_data': variation.meta_data,
} for variation in item.variations.all() } for variation in item.variations.all()
] ]
} for item in self.event.items.select_related('tax_rule').prefetch_related( } for item in self.event.items.select_related('tax_rule').prefetch_related('variations')
Prefetch(
'meta_values',
ItemMetaValue.objects.select_related('property'),
to_attr='meta_values_cached'
),
Prefetch(
'variations',
queryset=ItemVariation.objects.prefetch_related(
Prefetch(
'meta_values',
ItemVariationMetaValue.objects.select_related('property'),
to_attr='meta_values_cached'
),
),
),
)
], ],
'questions': [ 'questions': [
{ {

View File

@@ -36,7 +36,7 @@ from collections import OrderedDict
from django import forms from django import forms
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _
from pretix.base.models import OrderPosition from pretix.base.models import OrderPosition
@@ -50,8 +50,6 @@ from ..signals import (
class MailExporter(BaseExporter): class MailExporter(BaseExporter):
identifier = 'mailaddrs' identifier = 'mailaddrs'
verbose_name = _('Email addresses (text file)') verbose_name = _('Email addresses (text file)')
category = pgettext_lazy('export_category', 'Order data')
description = _("Download a text file with all email addresses collected either from buyers or from ticket holders.")
def render(self, form_data: dict): def render(self, form_data: dict):
qs = Order.objects.filter(event__in=self.events, status__in=form_data['status']).prefetch_related('event') qs = Order.objects.filter(event__in=self.events, status__in=form_data['status']).prefetch_related('event')

View File

@@ -33,8 +33,10 @@
# License for the specific language governing permissions and limitations under the License. # License for the specific language governing permissions and limitations under the License.
from collections import OrderedDict from collections import OrderedDict
from datetime import date, datetime, time
from decimal import Decimal from decimal import Decimal
import dateutil
import pytz import pytz
from django import forms from django import forms
from django.db.models import ( from django.db.models import (
@@ -44,10 +46,8 @@ from django.db.models import (
from django.db.models.functions import Coalesce from django.db.models.functions import Coalesce
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.functional import cached_property from django.utils.functional import cached_property
from django.utils.timezone import get_current_timezone, now from django.utils.timezone import get_current_timezone, make_aware, now
from django.utils.translation import ( from django.utils.translation import gettext as _, gettext_lazy, pgettext
gettext as _, gettext_lazy, pgettext, pgettext_lazy,
)
from pretix.base.models import ( from pretix.base.models import (
GiftCard, GiftCardTransaction, Invoice, InvoiceAddress, Order, GiftCard, GiftCardTransaction, Invoice, InvoiceAddress, Order,
@@ -63,24 +63,14 @@ from ...helpers.iter import chunked_iterable
from ..exporter import ( from ..exporter import (
ListExporter, MultiSheetListExporter, OrganizerLevelExportMixin, ListExporter, MultiSheetListExporter, OrganizerLevelExportMixin,
) )
from ..forms.widgets import SplitDateTimePickerWidget
from ..signals import ( from ..signals import (
register_data_exporters, register_multievent_data_exporters, register_data_exporters, register_multievent_data_exporters,
) )
from ..timeframes import (
DateFrameField,
resolve_timeframe_to_datetime_start_inclusive_end_exclusive,
)
class OrderListExporter(MultiSheetListExporter): class OrderListExporter(MultiSheetListExporter):
identifier = 'orderlist' identifier = 'orderlist'
verbose_name = gettext_lazy('Order data') verbose_name = gettext_lazy('Order data')
category = pgettext_lazy('export_category', 'Order data')
description = gettext_lazy('Download a spreadsheet of all orders. The spreadsheet will include three sheets, one '
'with a line for every order, one with a line for every order position, and one with '
'a line for every additional fee charged in an order.')
featured = True
@cached_property @cached_property
def providers(self): def providers(self):
@@ -115,25 +105,41 @@ class OrderListExporter(MultiSheetListExporter):
initial=False, initial=False,
required=False required=False
)), )),
('date_range', ('date_from',
DateFrameField( forms.DateField(
label=_('Date range'), label=_('Start date'),
include_future_frames=False, widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False, required=False,
help_text=_('Only include orders created within this date range.') help_text=_('Only include orders created on or after this date.')
)), )),
('event_date_range', ('date_to',
DateFrameField( forms.DateField(
label=_('Event date'), label=_('End date'),
include_future_frames=True, widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False, required=False,
help_text=_('Only include orders including at least one ticket for a date in this range. ' help_text=_('Only include orders created on or before this date.')
)),
('event_date_from',
forms.DateField(
label=_('Start event date'),
widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False,
help_text=_('Only include orders including at least one ticket for a date on or after this date. '
'Will also include other dates in case of mixed orders!')
)),
('event_date_to',
forms.DateField(
label=_('End event date'),
widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False,
help_text=_('Only include orders including at least one ticket for a date on or before this date. '
'Will also include other dates in case of mixed orders!') 'Will also include other dates in case of mixed orders!')
)), )),
] ]
d = OrderedDict(d) d = OrderedDict(d)
if not self.is_multievent and not self.event.has_subevents: if not self.is_multievent and not self.event.has_subevents:
del d['event_date_range'] del d['event_date_from']
del d['event_date_to']
return d return d
def _get_all_payment_methods(self, qs): def _get_all_payment_methods(self, qs):
@@ -176,27 +182,45 @@ class OrderListExporter(MultiSheetListExporter):
annotations = {} annotations = {}
filters = {} filters = {}
if form_data.get('date_range'): if form_data.get('date_from'):
dt_start, dt_end = resolve_timeframe_to_datetime_start_inclusive_end_exclusive(now(), form_data['date_range'], self.timezone) date_value = form_data.get('date_from')
if dt_start: if not isinstance(date_value, date):
filters[f'{rel}datetime__gte'] = dt_start date_value = dateutil.parser.parse(date_value).date()
if dt_end: datetime_value = make_aware(datetime.combine(date_value, time(0, 0, 0)), self.timezone)
filters[f'{rel}datetime__lt'] = dt_end
if form_data.get('event_date_range'): filters[f'{rel}datetime__gte'] = datetime_value
dt_start, dt_end = resolve_timeframe_to_datetime_start_inclusive_end_exclusive(now(), form_data['event_date_range'], self.timezone)
if dt_start: if form_data.get('date_to'):
annotations['event_date_max'] = Case( date_value = form_data.get('date_to')
When(**{f'{rel}event__has_subevents': True}, then=Max(f'{rel}all_positions__subevent__date_from')), if not isinstance(date_value, date):
default=F(f'{rel}event__date_from'), date_value = dateutil.parser.parse(date_value).date()
) datetime_value = make_aware(datetime.combine(date_value, time(23, 59, 59, 999999)), self.timezone)
filters['event_date_max__gte'] = dt_start
if dt_end: filters[f'{rel}datetime__lte'] = datetime_value
annotations['event_date_min'] = Case(
When(**{f'{rel}event__has_subevents': True}, then=Min(f'{rel}all_positions__subevent__date_from')), if form_data.get('event_date_from'):
default=F(f'{rel}event__date_from'), date_value = form_data.get('event_date_from')
) if not isinstance(date_value, date):
filters['event_date_min__lt'] = dt_end date_value = dateutil.parser.parse(date_value).date()
datetime_value = make_aware(datetime.combine(date_value, time(0, 0, 0)), self.timezone)
annotations['event_date_max'] = Case(
When(**{f'{rel}event__has_subevents': True}, then=Max(f'{rel}all_positions__subevent__date_from')),
default=F(f'{rel}event__date_from'),
)
filters['event_date_max__gte'] = datetime_value
if form_data.get('event_date_to'):
date_value = form_data.get('event_date_to')
if not isinstance(date_value, date):
date_value = dateutil.parser.parse(date_value).date()
datetime_value = make_aware(datetime.combine(date_value, time(23, 59, 59, 999999)), self.timezone)
annotations['event_date_min'] = Case(
When(**{f'{rel}event__has_subevents': True}, then=Min(f'{rel}all_positions__subevent__date_from')),
default=F(f'{rel}event__date_from'),
)
filters['event_date_min__lte'] = datetime_value
if filters: if filters:
return qs.annotate(**annotations).filter(**filters) return qs.annotate(**annotations).filter(**filters)
@@ -279,8 +303,6 @@ class OrderListExporter(MultiSheetListExporter):
for id, vn in payment_methods: for id, vn in payment_methods:
headers.append(_('Paid by {method}').format(method=vn)) headers.append(_('Paid by {method}').format(method=vn))
# get meta_data labels from first cached event
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
yield headers yield headers
full_fee_sum_cache = { full_fee_sum_cache = {
@@ -394,7 +416,6 @@ class OrderListExporter(MultiSheetListExporter):
payment_sum_cache.get((order.id, id), Decimal('0.00')) - payment_sum_cache.get((order.id, id), Decimal('0.00')) -
refund_sum_cache.get((order.id, id), Decimal('0.00')) refund_sum_cache.get((order.id, id), Decimal('0.00'))
) )
row += self.event_object_cache[order.event_id].meta_data.values()
yield row yield row
def iterate_fees(self, form_data: dict): def iterate_fees(self, form_data: dict):
@@ -444,9 +465,6 @@ class OrderListExporter(MultiSheetListExporter):
headers.append(_('External customer ID')) headers.append(_('External customer ID'))
headers.append(_('Payment providers')) headers.append(_('Payment providers'))
# get meta_data labels from first cached event
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
yield headers yield headers
yield self.ProgressSetTotal(total=qs.count()) yield self.ProgressSetTotal(total=qs.count())
@@ -494,7 +512,6 @@ class OrderListExporter(MultiSheetListExporter):
str(self.providers.get(p, p)) for p in sorted(set((op.payment_providers or '').split(','))) str(self.providers.get(p, p)) for p in sorted(set((op.payment_providers or '').split(',')))
if p and p != 'free' if p and p != 'free'
])) ]))
row += self.event_object_cache[order.event_id].meta_data.values()
yield row yield row
def iterate_positions(self, form_data: dict): def iterate_positions(self, form_data: dict):
@@ -516,7 +533,6 @@ class OrderListExporter(MultiSheetListExporter):
'order', 'order__invoice_address', 'order__customer', 'item', 'variation', 'order', 'order__invoice_address', 'order__customer', 'item', 'variation',
'voucher', 'tax_rule' 'voucher', 'tax_rule'
).prefetch_related( ).prefetch_related(
'subevent', 'subevent__meta_values',
'answers', 'answers__question', 'answers__options' 'answers', 'answers__question', 'answers__options'
) )
if form_data['paid_only']: if form_data['paid_only']:
@@ -608,10 +624,6 @@ class OrderListExporter(MultiSheetListExporter):
_('Payment providers'), _('Payment providers'),
] ]
# get meta_data labels from first cached event
meta_data_labels = next(iter(self.event_object_cache.values())).meta_data.keys()
if has_subevents:
headers += meta_data_labels
yield headers yield headers
all_ids = list(base_qs.order_by('order__datetime', 'positionid').values_list('pk', flat=True)) all_ids = list(base_qs.order_by('order__datetime', 'positionid').values_list('pk', flat=True))
@@ -735,12 +747,6 @@ class OrderListExporter(MultiSheetListExporter):
str(self.providers.get(p, p)) for p in sorted(set((op.payment_providers or '').split(','))) str(self.providers.get(p, p)) for p in sorted(set((op.payment_providers or '').split(',')))
if p and p != 'free' if p and p != 'free'
])) ]))
if has_subevents:
if op.subevent:
row += op.subevent.meta_data.values()
else:
row += [''] * len(meta_data_labels)
yield row yield row
def get_filename(self): def get_filename(self):
@@ -752,10 +758,7 @@ class OrderListExporter(MultiSheetListExporter):
class PaymentListExporter(ListExporter): class PaymentListExporter(ListExporter):
identifier = 'paymentlist' identifier = 'paymentlist'
verbose_name = gettext_lazy('Payments and refunds') verbose_name = gettext_lazy('Order payments and refunds')
category = pgettext_lazy('export_category', 'Order data')
description = gettext_lazy('Download a spreadsheet of all payments or refunds of every order.')
featured = True
@property @property
def additional_form_fields(self): def additional_form_fields(self):
@@ -834,8 +837,6 @@ class PaymentListExporter(ListExporter):
class QuotaListExporter(ListExporter): class QuotaListExporter(ListExporter):
identifier = 'quotalist' identifier = 'quotalist'
verbose_name = gettext_lazy('Quota availabilities') verbose_name = gettext_lazy('Quota availabilities')
category = pgettext_lazy('export_category', 'Product data')
description = gettext_lazy('Download a spreadsheet of all quotas including their current availability.')
def iterate_list(self, form_data): def iterate_list(self, form_data):
has_subevents = self.event.has_subevents has_subevents = self.event.has_subevents
@@ -889,17 +890,21 @@ class GiftcardTransactionListExporter(OrganizerLevelExportMixin, ListExporter):
identifier = 'giftcardtransactionlist' identifier = 'giftcardtransactionlist'
verbose_name = gettext_lazy('Gift card transactions') verbose_name = gettext_lazy('Gift card transactions')
organizer_required_permission = 'can_manage_gift_cards' organizer_required_permission = 'can_manage_gift_cards'
category = pgettext_lazy('export_category', 'Gift cards')
description = gettext_lazy('Download a spreadsheet of all gift card transactions.')
@property @property
def additional_form_fields(self): def additional_form_fields(self):
d = [ d = [
('date_range', ('date_from',
DateFrameField( forms.DateField(
label=_('Date range'), label=_('Start date'),
include_future_frames=False, widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False required=False,
)),
('date_to',
forms.DateField(
label=_('End date'),
widget=forms.DateInput(attrs={'class': 'datepickerfield'}),
required=False,
)), )),
] ]
d = OrderedDict(d) d = OrderedDict(d)
@@ -910,12 +915,22 @@ class GiftcardTransactionListExporter(OrganizerLevelExportMixin, ListExporter):
card__issuer=self.organizer, card__issuer=self.organizer,
).order_by('datetime').select_related('card', 'order', 'order__event') ).order_by('datetime').select_related('card', 'order', 'order__event')
if form_data.get('date_range'): if form_data.get('date_from'):
dt_start, dt_end = resolve_timeframe_to_datetime_start_inclusive_end_exclusive(now(), form_data['date_range'], self.timezone) date_value = form_data.get('date_from')
if dt_start: if isinstance(date_value, str):
qs = qs.filter(datetime__gte=dt_start) date_value = dateutil.parser.parse(date_value).date()
if dt_end: qs = qs.filter(
qs = qs.filter(datetime__lt=dt_end) datetime__gte=make_aware(datetime.combine(date_value, time(0, 0, 0)), self.timezone)
)
if form_data.get('date_to'):
date_value = form_data.get('date_to')
if isinstance(date_value, str):
date_value = dateutil.parser.parse(date_value).date()
qs = qs.filter(
datetime__lte=make_aware(datetime.combine(date_value, time(23, 59, 59, 999999)), self.timezone)
)
headers = [ headers = [
_('Gift card code'), _('Gift card code'),
@@ -945,8 +960,6 @@ class GiftcardTransactionListExporter(OrganizerLevelExportMixin, ListExporter):
class GiftcardRedemptionListExporter(ListExporter): class GiftcardRedemptionListExporter(ListExporter):
identifier = 'giftcardredemptionlist' identifier = 'giftcardredemptionlist'
verbose_name = gettext_lazy('Gift card redemptions') verbose_name = gettext_lazy('Gift card redemptions')
category = pgettext_lazy('export_category', 'Order data')
description = gettext_lazy('Download a spreadsheet of all payments or refunds that involve gift cards.')
def iterate_list(self, form_data): def iterate_list(self, form_data):
payments = OrderPayment.objects.filter( payments = OrderPayment.objects.filter(
@@ -992,18 +1005,14 @@ class GiftcardListExporter(OrganizerLevelExportMixin, ListExporter):
identifier = 'giftcardlist' identifier = 'giftcardlist'
verbose_name = gettext_lazy('Gift cards') verbose_name = gettext_lazy('Gift cards')
organizer_required_permission = 'can_manage_gift_cards' organizer_required_permission = 'can_manage_gift_cards'
category = pgettext_lazy('export_category', 'Gift cards')
description = gettext_lazy('Download a spreadsheet of all gift cards including their current value.')
@property @property
def additional_form_fields(self): def additional_form_fields(self):
return OrderedDict( return OrderedDict(
[ [
('date', forms.SplitDateTimeField( ('date', forms.DateTimeField(
label=_('Show value at'), label=_('Show value at'),
required=False, initial=now(),
widget=SplitDateTimePickerWidget(),
help_text=_('Defaults to the time of report.')
)), )),
('testmode', forms.ChoiceField( ('testmode', forms.ChoiceField(
label=_('Test mode'), label=_('Test mode'),
@@ -1031,13 +1040,12 @@ class GiftcardListExporter(OrganizerLevelExportMixin, ListExporter):
) )
def iterate_list(self, form_data): def iterate_list(self, form_data):
d = form_data.get('date') or now()
s = GiftCardTransaction.objects.filter( s = GiftCardTransaction.objects.filter(
card=OuterRef('pk'), card=OuterRef('pk'),
datetime__lte=d datetime__lte=form_data['date']
).order_by().values('card').annotate(s=Sum('value')).values('s') ).order_by().values('card').annotate(s=Sum('value')).values('s')
qs = self.organizer.issued_gift_cards.filter( qs = self.organizer.issued_gift_cards.filter(
issuance__lte=d issuance__lte=form_data['date']
).annotate( ).annotate(
cached_value=Coalesce(Subquery(s), Decimal('0.00')), cached_value=Coalesce(Subquery(s), Decimal('0.00')),
).order_by('issuance').prefetch_related( ).order_by('issuance').prefetch_related(
@@ -1052,11 +1060,11 @@ class GiftcardListExporter(OrganizerLevelExportMixin, ListExporter):
if form_data.get('state') == 'empty': if form_data.get('state') == 'empty':
qs = qs.filter(cached_value=0) qs = qs.filter(cached_value=0)
elif form_data.get('state') == 'valid_value': elif form_data.get('state') == 'valid_value':
qs = qs.exclude(cached_value=0).filter(Q(expires__isnull=True) | Q(expires__gte=d)) qs = qs.exclude(cached_value=0).filter(Q(expires__isnull=True) | Q(expires__gte=form_data['date']))
elif form_data.get('state') == 'expired_value': elif form_data.get('state') == 'expired_value':
qs = qs.exclude(cached_value=0).filter(expires__lt=d) qs = qs.exclude(cached_value=0).filter(expires__lt=form_data['date'])
elif form_data.get('state') == 'expired': elif form_data.get('state') == 'expired':
qs = qs.filter(expires__lt=d) qs = qs.filter(expires__lt=form_data['date'])
headers = [ headers = [
_('Gift card code'), _('Gift card code'),

View File

@@ -39,8 +39,6 @@ from ..signals import (
class WaitingListExporter(ListExporter): class WaitingListExporter(ListExporter):
identifier = 'waitinglist' identifier = 'waitinglist'
verbose_name = _('Waiting list') verbose_name = _('Waiting list')
category = pgettext_lazy('export_category', 'Waiting list')
description = _('Download a spread sheet with all your waiting list data.')
# map selected status to label and queryset-filter # map selected status to label and queryset-filter
status_filters = [ status_filters = [

View File

@@ -135,10 +135,6 @@ class NamePartsWidget(forms.MultiWidget):
data.append(value.get(fname, "")) data.append(value.get(fname, ""))
if '_legacy' in value and not data[-1]: if '_legacy' in value and not data[-1]:
data[-1] = value.get('_legacy', '') data[-1] = value.get('_legacy', '')
elif not any(d for d in data) and '_scheme' in value:
scheme = PERSON_NAME_SCHEMES[value['_scheme']]
data[-1] = scheme['concatenation'](value).strip()
return data return data
def render(self, name: str, value, attrs=None, renderer=None) -> str: def render(self, name: str, value, attrs=None, renderer=None) -> str:
@@ -531,7 +527,7 @@ class PortraitImageField(SizeValidationMixin, ExtValidationMixin, forms.FileFiel
code='aspect_ratio_not_3_by_4', code='aspect_ratio_not_3_by_4',
) )
except Exception as exc: except Exception as exc:
logger.exception('Could not parse image') logger.exception('foo')
# Pillow doesn't recognize it as an image. # Pillow doesn't recognize it as an image.
if isinstance(exc, ValidationError): if isinstance(exc, ValidationError):
raise raise
@@ -575,7 +571,7 @@ class BaseQuestionsForm(forms.Form):
add_fields = {} add_fields = {}
if item.ask_attendee_data and event.settings.attendee_names_asked: if item.admission and event.settings.attendee_names_asked:
add_fields['attendee_name_parts'] = NamePartsFormField( add_fields['attendee_name_parts'] = NamePartsFormField(
max_length=255, max_length=255,
required=event.settings.attendee_names_required and not self.all_optional, required=event.settings.attendee_names_required and not self.all_optional,
@@ -584,7 +580,7 @@ class BaseQuestionsForm(forms.Form):
label=_('Attendee name'), label=_('Attendee name'),
initial=(cartpos.attendee_name_parts if cartpos else orderpos.attendee_name_parts), initial=(cartpos.attendee_name_parts if cartpos else orderpos.attendee_name_parts),
) )
if item.ask_attendee_data and event.settings.attendee_emails_asked: if item.admission and event.settings.attendee_emails_asked:
add_fields['attendee_email'] = forms.EmailField( add_fields['attendee_email'] = forms.EmailField(
required=event.settings.attendee_emails_required and not self.all_optional, required=event.settings.attendee_emails_required and not self.all_optional,
label=_('Attendee email'), label=_('Attendee email'),
@@ -595,7 +591,7 @@ class BaseQuestionsForm(forms.Form):
} }
) )
) )
if item.ask_attendee_data and event.settings.attendee_company_asked: if item.admission and event.settings.attendee_company_asked:
add_fields['company'] = forms.CharField( add_fields['company'] = forms.CharField(
required=event.settings.attendee_company_required and not self.all_optional, required=event.settings.attendee_company_required and not self.all_optional,
label=_('Company'), label=_('Company'),
@@ -603,7 +599,7 @@ class BaseQuestionsForm(forms.Form):
initial=(cartpos.company if cartpos else orderpos.company), initial=(cartpos.company if cartpos else orderpos.company),
) )
if item.ask_attendee_data and event.settings.attendee_addresses_asked: if item.admission and event.settings.attendee_addresses_asked:
add_fields['street'] = forms.CharField( add_fields['street'] = forms.CharField(
required=event.settings.attendee_addresses_required and not self.all_optional, required=event.settings.attendee_addresses_required and not self.all_optional,
label=_('Address'), label=_('Address'),
@@ -919,7 +915,6 @@ class BaseQuestionsForm(forms.Form):
class BaseInvoiceAddressForm(forms.ModelForm): class BaseInvoiceAddressForm(forms.ModelForm):
vat_warning = False vat_warning = False
address_validation = False
class Meta: class Meta:
model = InvoiceAddress model = InvoiceAddress
@@ -1055,9 +1050,6 @@ class BaseInvoiceAddressForm(forms.ModelForm):
v.widget.attrs['autocomplete'] = 'section-invoice billing ' + v.widget.attrs.get('autocomplete', '') v.widget.attrs['autocomplete'] = 'section-invoice billing ' + v.widget.attrs.get('autocomplete', '')
def clean(self): def clean(self):
from pretix.base.addressvalidation import \
validate_address # local import to prevent impact on startup time
data = self.cleaned_data data = self.cleaned_data
if not data.get('is_business'): if not data.get('is_business'):
data['company'] = '' data['company'] = ''
@@ -1073,8 +1065,9 @@ class BaseInvoiceAddressForm(forms.ModelForm):
if 'vat_id' in self.changed_data or not data.get('vat_id'): if 'vat_id' in self.changed_data or not data.get('vat_id'):
self.instance.vat_id_validated = False self.instance.vat_id_validated = False
if self.address_validation: if data.get('city') and data.get('country') and str(data['country']) in COUNTRIES_WITH_STATE_IN_ADDRESS:
self.cleaned_data = data = validate_address(data, self.all_optional) if not data.get('state'):
self.add_error('state', _('This field is required.'))
self.instance.name_parts = data.get('name_parts') self.instance.name_parts = data.get('name_parts')

View File

@@ -23,7 +23,6 @@ import logging
from collections import defaultdict from collections import defaultdict
from decimal import Decimal from decimal import Decimal
from io import BytesIO from io import BytesIO
from itertools import groupby
from typing import Tuple from typing import Tuple
import bleach import bleach
@@ -242,12 +241,6 @@ class BaseReportlabInvoiceRenderer(BaseInvoiceRenderer):
buffer.seek(0) buffer.seek(0)
return 'invoice.pdf', 'application/pdf', buffer.read() return 'invoice.pdf', 'application/pdf', buffer.read()
def _clean_text(self, text, tags=None):
return bleach.clean(
text,
tags=tags or []
).strip().replace('<br>', '<br />').replace('\n', '<br />\n')
class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer): class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
identifier = 'classic' identifier = 'classic'
@@ -272,7 +265,7 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
invoice_to_top = 52 * mm invoice_to_top = 52 * mm
def _draw_invoice_to(self, canvas): def _draw_invoice_to(self, canvas):
p = Paragraph(self._clean_text(self.invoice.address_invoice_to), p = Paragraph(bleach.clean(self.invoice.address_invoice_to, tags=[]).strip().replace('\n', '<br />\n'),
style=self.stylesheet['Normal']) style=self.stylesheet['Normal'])
p.wrapOn(canvas, self.invoice_to_width, self.invoice_to_height) p.wrapOn(canvas, self.invoice_to_width, self.invoice_to_height)
p_size = p.wrap(self.invoice_to_width, self.invoice_to_height) p_size = p.wrap(self.invoice_to_width, self.invoice_to_height)
@@ -285,7 +278,7 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
def _draw_invoice_from(self, canvas): def _draw_invoice_from(self, canvas):
p = Paragraph( p = Paragraph(
self._clean_text(self.invoice.full_invoice_from), bleach.clean(self.invoice.full_invoice_from, tags=[]).strip().replace('\n', '<br />\n'),
style=self.stylesheet['InvoiceFrom'] style=self.stylesheet['InvoiceFrom']
) )
p.wrapOn(canvas, self.invoice_from_width, self.invoice_from_height) p.wrapOn(canvas, self.invoice_from_width, self.invoice_from_height)
@@ -480,8 +473,8 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
if self.invoice.custom_field: if self.invoice.custom_field:
story.append(Paragraph( story.append(Paragraph(
'{}: {}'.format( '{}: {}'.format(
self._clean_text(str(self.invoice.event.settings.invoice_address_custom_field)), bleach.clean(str(self.invoice.event.settings.invoice_address_custom_field), tags=[]).strip().replace('\n', '<br />\n'),
self._clean_text(self.invoice.custom_field), bleach.clean(self.invoice.custom_field, tags=[]).strip().replace('\n', '<br />\n'),
), ),
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
@@ -489,7 +482,7 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
if self.invoice.internal_reference: if self.invoice.internal_reference:
story.append(Paragraph( story.append(Paragraph(
pgettext('invoice', 'Customer reference: {reference}').format( pgettext('invoice', 'Customer reference: {reference}').format(
reference=self._clean_text(self.invoice.internal_reference), reference=bleach.clean(self.invoice.internal_reference, tags=[]).strip().replace('\n', '<br />\n'),
), ),
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
@@ -497,20 +490,20 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
if self.invoice.invoice_to_vat_id: if self.invoice.invoice_to_vat_id:
story.append(Paragraph( story.append(Paragraph(
pgettext('invoice', 'Customer VAT ID') + ': ' + pgettext('invoice', 'Customer VAT ID') + ': ' +
self._clean_text(self.invoice.invoice_to_vat_id), bleach.clean(self.invoice.invoice_to_vat_id, tags=[]).replace("\n", "<br />\n"),
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
if self.invoice.invoice_to_beneficiary: if self.invoice.invoice_to_beneficiary:
story.append(Paragraph( story.append(Paragraph(
pgettext('invoice', 'Beneficiary') + ':<br />' + pgettext('invoice', 'Beneficiary') + ':<br />' +
self._clean_text(self.invoice.invoice_to_beneficiary), bleach.clean(self.invoice.invoice_to_beneficiary, tags=[]).replace("\n", "<br />\n"),
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
if self.invoice.introductory_text: if self.invoice.introductory_text:
story.append(Paragraph( story.append(Paragraph(
self._clean_text(self.invoice.introductory_text, tags=['br']), self.invoice.introductory_text,
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
story.append(Spacer(1, 10 * mm)) story.append(Spacer(1, 10 * mm))
@@ -561,47 +554,31 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
pgettext('invoice', 'Amount'), pgettext('invoice', 'Amount'),
)] )]
def _group_key(line):
return (line.description, line.tax_rate, line.tax_name, line.net_value, line.gross_value, line.subevent_id,
line.event_date_from, line.event_date_to)
total = Decimal('0.00') total = Decimal('0.00')
for (description, tax_rate, tax_name, net_value, gross_value, *ignored), lines in groupby(self.invoice.lines.all(), key=_group_key): for line in self.invoice.lines.all():
lines = list(lines)
if has_taxes: if has_taxes:
if len(lines) > 1:
single_price_line = pgettext('invoice', 'Single price: {net_price} net / {gross_price} gross').format(
net_price=money_filter(net_value, self.invoice.event.currency),
gross_price=money_filter(gross_value, self.invoice.event.currency),
)
description = description + "\n" + single_price_line
tdata.append(( tdata.append((
Paragraph( Paragraph(
self._clean_text(description, tags=['br']), bleach.clean(line.description, tags=['br']).strip().replace('<br>', '<br/>').replace('\n', '<br />\n'),
self.stylesheet['Normal'] self.stylesheet['Normal']
), ),
str(len(lines)), "1",
localize(tax_rate) + " %", localize(line.tax_rate) + " %",
money_filter(net_value * len(lines), self.invoice.event.currency), money_filter(line.net_value, self.invoice.event.currency),
money_filter(gross_value * len(lines), self.invoice.event.currency), money_filter(line.gross_value, self.invoice.event.currency),
)) ))
else: else:
if len(lines) > 1:
single_price_line = pgettext('invoice', 'Single price: {price}').format(
price=money_filter(gross_value, self.invoice.event.currency),
)
description = description + "\n" + single_price_line
tdata.append(( tdata.append((
Paragraph( Paragraph(
self._clean_text(description, tags=['br']), bleach.clean(line.description, tags=['br']).strip().replace('<br>', '<br/>').replace('\n', '<br />\n'),
self.stylesheet['Normal'] self.stylesheet['Normal']
), ),
str(len(lines)), "1",
money_filter(gross_value * len(lines), self.invoice.event.currency), money_filter(line.gross_value, self.invoice.event.currency),
)) ))
taxvalue_map[tax_rate, tax_name] += (gross_value - net_value) * len(lines) taxvalue_map[line.tax_rate, line.tax_name] += line.tax_value
grossvalue_map[tax_rate, tax_name] += gross_value * len(lines) grossvalue_map[line.tax_rate, line.tax_name] += line.gross_value
total += gross_value * len(lines) total += line.gross_value
if has_taxes: if has_taxes:
tdata.append([ tdata.append([
@@ -663,7 +640,7 @@ class ClassicInvoiceRenderer(BaseReportlabInvoiceRenderer):
if self.invoice.additional_text: if self.invoice.additional_text:
story.append(Paragraph( story.append(Paragraph(
self._clean_text(self.invoice.additional_text, tags=['br']), self.invoice.additional_text,
self.stylesheet['Normal'] self.stylesheet['Normal']
)) ))
story.append(Spacer(1, 5 * mm)) story.append(Spacer(1, 5 * mm))
@@ -800,7 +777,7 @@ class Modern1Renderer(ClassicInvoiceRenderer):
if not self.invoice.invoice_from: if not self.invoice.invoice_from:
return return
c = [ c = [
self._clean_text(l) bleach.clean(l, tags=[]).strip().replace('\n', '<br />\n')
for l in self.invoice.address_invoice_from.strip().split('\n') for l in self.invoice.address_invoice_from.strip().split('\n')
] ]
p = Paragraph(' · '.join(c), style=self.stylesheet['Sender']) p = Paragraph(' · '.join(c), style=self.stylesheet['Sender'])

View File

@@ -103,8 +103,6 @@ class Command(BaseCommand):
with language(locale), override(timezone): with language(locale), override(timezone):
for receiver, response in signal_result: for receiver, response in signal_result:
if not response:
return None
ex = response(e, o, report_status) ex = response(e, o, report_status)
if ex.identifier == options['export_provider']: if ex.identifier == options['export_provider']:
params = json.loads(options.get('parameters') or '{}') params = json.loads(options.get('parameters') or '{}')

View File

@@ -79,9 +79,9 @@ class Command(BaseCommand):
if settings.SENTRY_ENABLED: if settings.SENTRY_ENABLED:
from sentry_sdk import capture_exception from sentry_sdk import capture_exception
capture_exception(err) capture_exception(err)
self.stdout.write(self.style.ERROR(f'ERROR {name}: {str(err)}\n')) self.stdout.write(self.style.ERROR(f'ERROR runperiodic {str(err)}\n'))
else: else:
self.stdout.write(self.style.ERROR(f'ERROR {name}: {str(err)}\n')) self.stdout.write(self.style.ERROR(f'ERROR runperiodic {str(err)}\n'))
traceback.print_exc() traceback.print_exc()
else: else:
if options.get('verbosity') > 1: if options.get('verbosity') > 1:

View File

@@ -30,6 +30,7 @@ from django.urls import get_script_prefix
from django.utils import timezone, translation from django.utils import timezone, translation
from django.utils.cache import patch_vary_headers from django.utils.cache import patch_vary_headers
from django.utils.deprecation import MiddlewareMixin from django.utils.deprecation import MiddlewareMixin
from django.utils.translation import LANGUAGE_SESSION_KEY
from django.utils.translation.trans_real import ( from django.utils.translation.trans_real import (
check_for_language, get_supported_language_variant, language_code_re, check_for_language, get_supported_language_variant, language_code_re,
parse_accept_lang_header, parse_accept_lang_header,
@@ -127,7 +128,12 @@ def get_language_from_user_settings(request: HttpRequest) -> str:
return lang_code return lang_code
def get_language_from_cookie(request: HttpRequest) -> str: def get_language_from_session_or_cookie(request: HttpRequest) -> str:
if hasattr(request, 'session'):
lang_code = request.session.get(LANGUAGE_SESSION_KEY)
if lang_code in _supported and lang_code is not None and check_for_language(lang_code):
return lang_code
lang_code = request.COOKIES.get(settings.LANGUAGE_COOKIE_NAME) lang_code = request.COOKIES.get(settings.LANGUAGE_COOKIE_NAME)
try: try:
return get_supported_language_variant(lang_code) return get_supported_language_variant(lang_code)
@@ -181,14 +187,14 @@ def get_language_from_request(request: HttpRequest) -> str:
return ( return (
get_language_from_user_settings(request) get_language_from_user_settings(request)
or get_language_from_customer_settings(request) or get_language_from_customer_settings(request)
or get_language_from_cookie(request) or get_language_from_session_or_cookie(request)
or get_language_from_browser(request) or get_language_from_browser(request)
or get_language_from_event(request) or get_language_from_event(request)
or get_default_language() or get_default_language()
) )
else: else:
return ( return (
get_language_from_cookie(request) get_language_from_session_or_cookie(request)
or get_language_from_customer_settings(request) or get_language_from_customer_settings(request)
or get_language_from_user_settings(request) or get_language_from_user_settings(request)
or get_language_from_browser(request) or get_language_from_browser(request)
@@ -218,11 +224,6 @@ def _merge_csp(a, b):
if k not in a: if k not in a:
a[k] = b[k] a[k] = b[k]
for k, v in a.items():
if "'unsafe-inline'" in v:
# If we need unsafe-inline, drop any hashes or nonce as they will be ignored otherwise
a[k] = [i for i in v if not i.startswith("'nonce-") and not i.startswith("'sha-")]
class SecurityMiddleware(MiddlewareMixin): class SecurityMiddleware(MiddlewareMixin):
CSP_EXEMPT = ( CSP_EXEMPT = (
@@ -300,7 +301,7 @@ class SecurityMiddleware(MiddlewareMixin):
resp['Content-Security-Policy'] = _render_csp(h).format(static=staticdomain, dynamic=dynamicdomain, resp['Content-Security-Policy'] = _render_csp(h).format(static=staticdomain, dynamic=dynamicdomain,
media=mediadomain) media=mediadomain)
for k, v in h.items(): for k, v in h.items():
h[k] = sorted(set(' '.join(v).format(static=staticdomain, dynamic=dynamicdomain, media=mediadomain).split(' '))) h[k] = ' '.join(v).format(static=staticdomain, dynamic=dynamicdomain, media=mediadomain).split(' ')
resp['Content-Security-Policy'] = _render_csp(h) resp['Content-Security-Policy'] = _render_csp(h)
elif 'Content-Security-Policy' in resp: elif 'Content-Security-Policy' in resp:
del resp['Content-Security-Policy'] del resp['Content-Security-Policy']

View File

@@ -0,0 +1,23 @@
# Generated by Django 3.2.2 on 2022-10-19 09:50
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0222_alter_question_unique_together'),
]
operations = [
migrations.AddField(
model_name='transaction',
name='source_identifier',
field=models.CharField(db_index=True, max_length=190, null=True),
),
migrations.AddField(
model_name='transaction',
name='source_type',
field=models.CharField(db_index=True, max_length=190, null=True),
),
]

View File

@@ -1,18 +0,0 @@
# Generated by Django 3.2.12 on 2022-10-12 09:13
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0222_alter_question_unique_together'),
]
operations = [
migrations.AddField(
model_name='voucher',
name='min_usages',
field=models.PositiveIntegerField(default=1),
),
]

View File

@@ -1,18 +0,0 @@
# Generated by Django 3.2.16 on 2022-11-14 11:32
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0223_voucher_min_usages'),
]
operations = [
migrations.AddField(
model_name='eventmetaproperty',
name='filter_allowed',
field=models.BooleanField(default=True),
),
]

View File

@@ -1,18 +0,0 @@
# Generated by Django 3.2.16 on 2022-11-17 15:27
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0224_eventmetaproperty_filter_allowed'),
]
operations = [
migrations.AddField(
model_name='orderpayment',
name='process_initiated',
field=models.BooleanField(null=True),
),
]

View File

@@ -1,29 +0,0 @@
# Generated by Django 3.2.16 on 2022-12-09 10:06
import django.db.models.deletion
from django.db import migrations, models
import pretix.base.models.base
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0225_orderpayment_process_initiated'),
]
operations = [
migrations.CreateModel(
name='ItemVariationMetaValue',
fields=[
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False)),
('value', models.TextField()),
('property', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='variation_values', to='pretixbase.itemmetaproperty')),
('variation', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='meta_values', to='pretixbase.itemvariation')),
],
options={
'unique_together': {('variation', 'property')},
},
bases=(models.Model, pretix.base.models.base.LoggingMixin),
),
]

View File

@@ -1,27 +0,0 @@
# Generated by Django 3.2.16 on 2022-12-21 08:59
from django.db import migrations, models
def item_set_personalized(apps, schema_editor):
# We cannot really know if a position was bundled or an add-on, but we can at least guess
Item = apps.get_model("pretixbase", "Item")
Item.objects.filter(admission=True).update(personalized=True)
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0226_itemvariationmetavalue'),
]
operations = [
migrations.AddField(
model_name='item',
name='personalized',
field=models.BooleanField(default=False),
),
migrations.RunPython(
item_set_personalized,
migrations.RunPython.noop,
),
]

View File

@@ -1,68 +0,0 @@
# Generated by Django 3.2.16 on 2023-01-18 11:57
import django.db.models.deletion
from django.conf import settings
from django.db import migrations, models
import pretix.base.models.base
class Migration(migrations.Migration):
dependencies = [
('pretixbase', '0227_item_personalized'),
]
operations = [
migrations.CreateModel(
name='ScheduledOrganizerExport',
fields=[
('id', models.BigAutoField(primary_key=True, serialize=False)),
('export_identifier', models.CharField(max_length=190)),
('export_form_data', models.JSONField(default=dict)),
('locale', models.CharField(max_length=250)),
('mail_additional_recipients', models.TextField()),
('mail_additional_recipients_cc', models.TextField()),
('mail_additional_recipients_bcc', models.TextField()),
('mail_subject', models.CharField(max_length=250)),
('mail_template', models.TextField()),
('schedule_rrule', models.TextField(null=True)),
('schedule_rrule_time', models.TimeField()),
('schedule_next_run', models.DateTimeField(blank=True, null=True)),
('error_counter', models.IntegerField(default=0)),
('error_last_message', models.TextField(null=True)),
('timezone', models.CharField(default='UTC', max_length=100)),
('organizer', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='scheduled_exports', to='pretixbase.organizer')),
('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to=settings.AUTH_USER_MODEL)),
],
options={
'abstract': False,
},
bases=(models.Model, pretix.base.models.base.LoggingMixin),
),
migrations.CreateModel(
name='ScheduledEventExport',
fields=[
('id', models.BigAutoField(primary_key=True, serialize=False)),
('export_identifier', models.CharField(max_length=190)),
('export_form_data', models.JSONField(default=dict)),
('locale', models.CharField(max_length=250)),
('mail_additional_recipients', models.TextField()),
('mail_additional_recipients_cc', models.TextField()),
('mail_additional_recipients_bcc', models.TextField()),
('mail_subject', models.CharField(max_length=250)),
('mail_template', models.TextField()),
('schedule_rrule', models.TextField(null=True)),
('schedule_rrule_time', models.TimeField()),
('schedule_next_run', models.DateTimeField(blank=True, null=True)),
('error_counter', models.IntegerField(default=0)),
('error_last_message', models.TextField(null=True)),
('event', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='scheduled_exports', to='pretixbase.event')),
('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to=settings.AUTH_USER_MODEL)),
],
options={
'abstract': False,
},
bases=(models.Model, pretix.base.models.base.LoggingMixin),
),
]

View File

@@ -30,13 +30,12 @@ from .event import (
Event, Event_SettingsStore, EventLock, EventMetaProperty, EventMetaValue, Event, Event_SettingsStore, EventLock, EventMetaProperty, EventMetaValue,
SubEvent, SubEventMetaValue, generate_invite_token, SubEvent, SubEventMetaValue, generate_invite_token,
) )
from .exports import ScheduledEventExport, ScheduledOrganizerExport
from .giftcards import GiftCard, GiftCardAcceptance, GiftCardTransaction from .giftcards import GiftCard, GiftCardAcceptance, GiftCardTransaction
from .invoices import Invoice, InvoiceLine, invoice_filename from .invoices import Invoice, InvoiceLine, invoice_filename
from .items import ( from .items import (
Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaProperty, ItemMetaValue, Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaProperty, ItemMetaValue,
ItemVariation, ItemVariationMetaValue, Question, QuestionOption, Quota, ItemVariation, Question, QuestionOption, Quota, SubEventItem,
SubEventItem, SubEventItemVariation, itempicture_upload_to, SubEventItemVariation, itempicture_upload_to,
) )
from .log import LogEntry from .log import LogEntry
from .memberships import Membership, MembershipType from .memberships import Membership, MembershipType

View File

@@ -36,17 +36,13 @@ from datetime import timedelta
from django.conf import settings from django.conf import settings
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.db import models from django.db import models
from django.db.models import ( from django.db.models import Exists, F, Max, OuterRef, Q, Subquery
Count, Exists, F, Max, OuterRef, Q, Subquery, Value, Window,
)
from django.db.models.expressions import RawSQL
from django.utils.timezone import now from django.utils.timezone import now
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _, pgettext_lazy
from django_scopes import ScopedManager, scopes_disabled from django_scopes import ScopedManager, scopes_disabled
from pretix.base.models import LoggedModel from pretix.base.models import LoggedModel
from pretix.base.models.fields import MultiStringField from pretix.base.models.fields import MultiStringField
from pretix.helpers import PostgresWindowFrame
class CheckinList(LoggedModel): class CheckinList(LoggedModel):
@@ -99,109 +95,52 @@ class CheckinList(LoggedModel):
class Meta: class Meta:
ordering = ('subevent__date_from', 'name') ordering = ('subevent__date_from', 'name')
def positions_query(self, ignore_status=False): @property
def positions(self):
from . import Order, OrderPosition from . import Order, OrderPosition
qs = OrderPosition.all.filter( qs = OrderPosition.objects.filter(
order__event=self.event, order__event=self.event,
order__status__in=[Order.STATUS_PAID, Order.STATUS_PENDING] if self.include_pending else [
Order.STATUS_PAID],
) )
if not ignore_status:
qs = qs.filter(
canceled=False,
order__status__in=[Order.STATUS_PAID, Order.STATUS_PENDING] if self.include_pending else [Order.STATUS_PAID],
)
if self.subevent_id: if self.subevent_id:
qs = qs.filter(subevent_id=self.subevent_id) qs = qs.filter(subevent_id=self.subevent_id)
if not self.all_products: if not self.all_products:
qs = qs.filter(item__in=self.limit_products.values_list('id', flat=True)) qs = qs.filter(item__in=self.limit_products.values_list('id', flat=True))
return qs return qs
@property
def positions(self):
return self.positions_query(ignore_status=False)
@scopes_disabled()
def positions_inside_query(self, ignore_status=False, at_time=None):
if at_time is None:
c_q = []
else:
c_q = [Q(datetime__lt=at_time)]
if "postgresql" not in settings.DATABASES["default"]["ENGINE"]:
# Use a simple approach that works on all databases
qs = self.positions_query(ignore_status=ignore_status).annotate(
last_entry=Subquery(
Checkin.objects.filter(
*c_q,
position_id=OuterRef('pk'),
list_id=self.pk,
type=Checkin.TYPE_ENTRY,
).order_by().values('position_id').annotate(
m=Max('datetime')
).values('m')
),
last_exit=Subquery(
Checkin.objects.filter(
*c_q,
position_id=OuterRef('pk'),
list_id=self.pk,
type=Checkin.TYPE_EXIT,
).order_by().values('position_id').annotate(
m=Max('datetime')
).values('m')
),
).filter(
Q(last_entry__isnull=False)
& Q(
Q(last_exit__isnull=True) | Q(last_exit__lt=F('last_entry'))
)
)
return qs
# Use the PostgreSQL-specific query using Window functions, which is a lot faster.
# On a real-world example with ~100k tickets, of which ~17k are checked in, we observed
# a speed-up from 29s (old) to a few hundred milliseconds (new)!
# Why is this so much faster? The regular query get's PostgreSQL all busy with filtering
# the tickets both by their belonging the event and checkin status at the same time, while
# this query just iterates over all successful checkins on the list, and -- by the power
# of window functions -- asks "is this an entry that is followed by no exit?". Then we
# dedupliate by position and count it up.
cl = self
base_q, base_params = (
Checkin.all.filter(*c_q, successful=True, list=cl)
.annotate(
cnt_exists_after=Window(
expression=Count("position_id", filter=Q(type=Value("exit"))),
partition_by=[F("position_id"), F("list_id")],
order_by=F("datetime").asc(),
frame=PostgresWindowFrame(
"ROWS", start="1 following", end="unbounded following"
),
)
)
.values("position_id", "type", "datetime", "cnt_exists_after")
.query.sql_with_params()
)
return self.positions_query(ignore_status=ignore_status).filter(
pk__in=RawSQL(
f"""
SELECT "position_id"
FROM ({str(base_q)}) s
WHERE "type" = %s AND "cnt_exists_after" = 0
GROUP BY "position_id"
""",
[*base_params, Checkin.TYPE_ENTRY]
)
)
@property @property
def positions_inside(self): def positions_inside(self):
return self.positions_inside_query(None) return self.positions.annotate(
last_entry=Subquery(
Checkin.objects.filter(
position_id=OuterRef('pk'),
list_id=self.pk,
type=Checkin.TYPE_ENTRY,
).order_by().values('position_id').annotate(
m=Max('datetime')
).values('m')
),
last_exit=Subquery(
Checkin.objects.filter(
position_id=OuterRef('pk'),
list_id=self.pk,
type=Checkin.TYPE_EXIT,
).order_by().values('position_id').annotate(
m=Max('datetime')
).values('m')
),
).filter(
Q(last_entry__isnull=False)
& Q(
Q(last_exit__isnull=True) | Q(last_exit__lt=F('last_entry'))
)
)
@property @property
def inside_count(self): def inside_count(self):
return self.positions_inside_query(None).count() return self.positions_inside.count()
@property @property
@scopes_disabled() @scopes_disabled()

View File

@@ -262,7 +262,7 @@ class Customer(LoggedModel):
) + '?id=' + self.identifier + '&token=' + token ) + '?id=' + self.identifier + '&token=' + token
mail( mail(
self.email, self.email,
self.organizer.settings.mail_subject_customer_registration, _('Activate your account at {organizer}').format(organizer=self.organizer.name),
self.organizer.settings.mail_text_customer_registration, self.organizer.settings.mail_text_customer_registration,
ctx, ctx,
locale=self.locale, locale=self.locale,

View File

@@ -28,7 +28,6 @@ from typing import Dict, Optional, Tuple
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.core.validators import MinValueValidator from django.core.validators import MinValueValidator
from django.db import models from django.db import models
from django.utils.timezone import now
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _, pgettext_lazy
from django_scopes import ScopedManager from django_scopes import ScopedManager
@@ -199,14 +198,6 @@ class Discount(LoggedModel):
'subevent_mode': self.subevent_mode, 'subevent_mode': self.subevent_mode,
}) })
def is_available_by_time(self, now_dt=None) -> bool:
now_dt = now_dt or now()
if self.available_from and self.available_from > now_dt:
return False
if self.available_until and self.available_until < now_dt:
return False
return True
def _apply_min_value(self, positions, idx_group, result): def _apply_min_value(self, positions, idx_group, result):
if self.condition_min_value and sum(positions[idx][2] for idx in idx_group) < self.condition_min_value: if self.condition_min_value and sum(positions[idx][2] for idx in idx_group) < self.condition_min_value:
return return

View File

@@ -374,7 +374,7 @@ class EventMixin:
if q.active_items: if q.active_items:
items_reserved.update(q.active_items.split(",")) items_reserved.update(q.active_items.split(","))
if q.active_variations: if q.active_variations:
vars_reserved.update(q.active_variations.split(",")) vars_available.update(q.active_variations.split(","))
elif res[0] < Quota.AVAILABILITY_RESERVED: elif res[0] < Quota.AVAILABILITY_RESERVED:
if q.active_items: if q.active_items:
items_gone.update(q.active_items.split(",")) items_gone.update(q.active_items.split(","))
@@ -590,7 +590,6 @@ class Event(EventMixin, LoggedModel):
self.settings.event_list_type = 'calendar' self.settings.event_list_type = 'calendar'
self.settings.invoice_email_attachment = True self.settings.invoice_email_attachment = True
self.settings.name_scheme = 'given_family' self.settings.name_scheme = 'given_family'
self.settings.payment_banktransfer_invoice_immediately = True
@property @property
def social_image(self): def social_image(self):
@@ -632,7 +631,6 @@ class Event(EventMixin, LoggedModel):
return super().presale_has_ended return super().presale_has_ended
def delete_all_orders(self, really=False): def delete_all_orders(self, really=False):
from .checkin import Checkin
from .orders import ( from .orders import (
OrderFee, OrderPayment, OrderPosition, OrderRefund, Transaction, OrderFee, OrderPayment, OrderPosition, OrderRefund, Transaction,
) )
@@ -646,7 +644,6 @@ class Event(EventMixin, LoggedModel):
OrderFee.objects.filter(order__event=self).delete() OrderFee.objects.filter(order__event=self).delete()
OrderRefund.objects.filter(order__event=self).delete() OrderRefund.objects.filter(order__event=self).delete()
OrderPayment.objects.filter(order__event=self).delete() OrderPayment.objects.filter(order__event=self).delete()
Checkin.objects.filter(list__event=self).delete()
self.orders.all().delete() self.orders.all().delete()
def save(self, *args, **kwargs): def save(self, *args, **kwargs):
@@ -730,7 +727,7 @@ class Event(EventMixin, LoggedModel):
from ..signals import event_copy_data from ..signals import event_copy_data
from . import ( from . import (
Discount, Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaValue, Discount, Item, ItemAddOn, ItemBundle, ItemCategory, ItemMetaValue,
ItemVariationMetaValue, Question, Quota, Question, Quota,
) )
# Note: avoid self.set_active_plugins(), it causes trouble e.g. for the badges plugin. # Note: avoid self.set_active_plugins(), it causes trouble e.g. for the badges plugin.
@@ -806,18 +803,12 @@ class Event(EventMixin, LoggedModel):
v.item = i v.item = i
v.save(force_insert=True) v.save(force_insert=True)
for imv in ItemMetaValue.objects.filter(item__event=other): for imv in ItemMetaValue.objects.filter(item__event=other).prefetch_related('item', 'property'):
imv.pk = None imv.pk = None
imv.property = item_meta_properties_map[imv.property_id] imv.property = item_meta_properties_map[imv.property.pk]
imv.item = item_map[imv.item.pk] imv.item = item_map[imv.item.pk]
imv.save(force_insert=True) imv.save(force_insert=True)
for imv in ItemVariationMetaValue.objects.filter(variation__item__event=other):
imv.pk = None
imv.property = item_meta_properties_map[imv.property_id]
imv.variation = variation_map[imv.variation_id]
imv.save(force_insert=True)
for ia in ItemAddOn.objects.filter(base_item__event=other).prefetch_related('base_item', 'addon_category'): for ia in ItemAddOn.objects.filter(base_item__event=other).prefetch_related('base_item', 'addon_category'):
ia.pk = None ia.pk = None
ia.base_item = item_map[ia.base_item.pk] ia.base_item = item_map[ia.base_item.pk]
@@ -1588,11 +1579,6 @@ class EventMetaProperty(LoggedModel):
verbose_name=_("Valid values"), verbose_name=_("Valid values"),
help_text=_("If you keep this empty, any value is allowed. Otherwise, enter one possible value per line.") help_text=_("If you keep this empty, any value is allowed. Otherwise, enter one possible value per line.")
) )
filter_allowed = models.BooleanField(
default=True, verbose_name=_("Can be used for filtering"),
help_text=_("This field will be shown to filter events or reports in the backend, and it can also be used "
"for hidden filter parameters in the frontend (e.g. using the widget).")
)
def full_clean(self, exclude=None, validate_unique=True): def full_clean(self, exclude=None, validate_unique=True):
super().full_clean(exclude, validate_unique) super().full_clean(exclude, validate_unique)

View File

@@ -1,139 +0,0 @@
#
# This file is part of pretix (Community Edition).
#
# Copyright (C) 2014-2020 Raphael Michel and contributors
# Copyright (C) 2020-2021 rami.io GmbH and contributors
#
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
# Public License as published by the Free Software Foundation in version 3 of the License.
#
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
# this file, see <https://pretix.eu/about/en/license>.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
# <https://www.gnu.org/licenses/>.
#
from datetime import datetime, timedelta
import pytz
from dateutil.rrule import rrulestr
from django.conf import settings
from django.core.serializers.json import DjangoJSONEncoder
from django.db import models
from django.utils.timezone import make_aware, now
from django.utils.translation import gettext_lazy as _
from pretix.base.models import LoggedModel
from pretix.base.validators import RRuleValidator, multimail_validate
class AbstractScheduledExport(LoggedModel):
id = models.BigAutoField(primary_key=True)
export_identifier = models.CharField(
max_length=190,
verbose_name=_("Export"),
)
export_form_data = models.JSONField(
default=dict,
encoder=DjangoJSONEncoder,
)
owner = models.ForeignKey(
"pretixbase.User",
on_delete=models.PROTECT,
)
locale = models.CharField(
verbose_name=_('Language'),
max_length=250
)
mail_additional_recipients = models.TextField(
verbose_name=_('Additional recipients'),
null=False, blank=True, validators=[multimail_validate],
help_text=_("You can specify multiple recipients separated by commas.")
)
mail_additional_recipients_cc = models.TextField(
verbose_name=_('Additional recipients (Cc)'),
null=False, blank=True, validators=[multimail_validate],
help_text=_("You can specify multiple recipients separated by commas.")
)
mail_additional_recipients_bcc = models.TextField(
verbose_name=_('Additional recipients (Bcc)'),
null=False, blank=True, validators=[multimail_validate],
help_text=_("You can specify multiple recipients separated by commas.")
)
mail_subject = models.CharField(
verbose_name=_('Subject'),
max_length=250
)
mail_template = models.TextField(
verbose_name=_('Message'),
)
schedule_rrule = models.TextField(
null=True, blank=True, validators=[RRuleValidator()]
)
schedule_rrule_time = models.TimeField(
verbose_name=_("Requested start time"),
help_text=_("The actual start time might be delayed depending on system load."),
)
schedule_next_run = models.DateTimeField(null=True, blank=True)
error_counter = models.IntegerField(default=0)
error_last_message = models.TextField(null=True, blank=True)
class Meta:
abstract = True
def __str__(self):
return self.mail_subject
def compute_next_run(self):
tz = self.tz
r = rrulestr(self.schedule_rrule)
base_dt = now().astimezone(tz).replace(tzinfo=None)
if now().astimezone(tz).time() < self.schedule_rrule_time:
base_dt -= timedelta(days=1)
new_d = r.after(base_dt, inc=False)
if not new_d:
self.schedule_next_run = None
return
try:
self.schedule_next_run = make_aware(datetime.combine(new_d.date(), self.schedule_rrule_time), tz)
except pytz.exceptions.AmbiguousTimeError:
self.schedule_next_run = make_aware(datetime.combine(new_d.date(), self.schedule_rrule_time), tz, is_dst=False)
except pytz.exceptions.NonExistentTimeError:
self.schedule_next_run = make_aware(datetime.combine(new_d.date(), self.schedule_rrule_time) + timedelta(hours=1), tz)
class ScheduledEventExport(AbstractScheduledExport):
event = models.ForeignKey(
"pretixbase.Event", on_delete=models.CASCADE, related_name="scheduled_exports"
)
@property
def tz(self):
return self.event.timezone
class ScheduledOrganizerExport(AbstractScheduledExport):
organizer = models.ForeignKey(
"pretixbase.Organizer", on_delete=models.CASCADE, related_name="scheduled_exports"
)
timezone = models.CharField(max_length=100,
default=settings.TIME_ZONE,
verbose_name=_('Timezone'))
@property
def tz(self):
return pytz.timezone(self.timezone)

View File

@@ -62,7 +62,6 @@ from pretix.base.models.base import LoggedModel
from pretix.base.models.fields import MultiStringField from pretix.base.models.fields import MultiStringField
from pretix.base.models.tax import TaxedPrice from pretix.base.models.tax import TaxedPrice
from ...helpers.images import ImageSizeValidator
from .event import Event, SubEvent from .event import Event, SubEvent
@@ -311,8 +310,6 @@ class Item(LoggedModel):
:type tax_rate: decimal.Decimal :type tax_rate: decimal.Decimal
:param admission: ``True``, if this item allows persons to enter the event (as opposed to e.g. merchandise) :param admission: ``True``, if this item allows persons to enter the event (as opposed to e.g. merchandise)
:type admission: bool :type admission: bool
:param personalized: ``True``, if attendee information should be collected for this ticket
:type personalized: bool
:param picture: A product picture to be shown next to the product description :param picture: A product picture to be shown next to the product description
:type picture: File :type picture: File
:param available_from: The date this product goes on sale :param available_from: The date this product goes on sale
@@ -399,14 +396,8 @@ class Item(LoggedModel):
admission = models.BooleanField( admission = models.BooleanField(
verbose_name=_("Is an admission ticket"), verbose_name=_("Is an admission ticket"),
help_text=_( help_text=_(
'Whether or not buying this product allows a person to enter your event' 'Whether or not buying this product allows a person to enter '
), 'your event'
default=False
)
personalized = models.BooleanField(
verbose_name=_("Is a personalized ticket"),
help_text=_(
'Whether or not buying this product allows to enter attendee information'
), ),
default=False default=False
) )
@@ -430,8 +421,7 @@ class Item(LoggedModel):
picture = models.ImageField( picture = models.ImageField(
verbose_name=_("Product picture"), verbose_name=_("Product picture"),
null=True, blank=True, max_length=255, null=True, blank=True, max_length=255,
upload_to=itempicture_upload_to, upload_to=itempicture_upload_to
validators=[ImageSizeValidator()]
) )
available_from = models.DateTimeField( available_from = models.DateTimeField(
verbose_name=_("Available from"), verbose_name=_("Available from"),
@@ -588,22 +578,21 @@ class Item(LoggedModel):
return self.event.settings.show_quota_left return self.event.settings.show_quota_left
return self.show_quota_left return self.show_quota_left
@property
def ask_attendee_data(self):
return self.admission and self.personalized
def tax(self, price=None, base_price_is='auto', currency=None, invoice_address=None, override_tax_rate=None, include_bundled=False): def tax(self, price=None, base_price_is='auto', currency=None, invoice_address=None, override_tax_rate=None, include_bundled=False):
price = price if price is not None else self.default_price price = price if price is not None else self.default_price
bundled_sum = Decimal('0.00') if not self.tax_rule:
bundled_sum_net = Decimal('0.00') t = TaxedPrice(gross=price, net=price, tax=Decimal('0.00'),
bundled_sum_tax = Decimal('0.00') rate=Decimal('0.00'), name='')
else:
t = self.tax_rule.tax(price, base_price_is=base_price_is, invoice_address=invoice_address,
override_tax_rate=override_tax_rate, currency=currency or self.event.currency)
if include_bundled: if include_bundled:
for b in self.bundles.all(): for b in self.bundles.all():
if b.designated_price and b.bundled_item.tax_rule_id != self.tax_rule_id: if b.designated_price and b.bundled_item.tax_rule_id != self.tax_rule_id:
if b.bundled_variation: if b.bundled_variation:
bprice = b.bundled_variation.tax(b.designated_price * b.count, bprice = b.bundled_variation.tax(b.designated_price * b.count, base_price_is='gross',
base_price_is='gross',
invoice_address=invoice_address, invoice_address=invoice_address,
currency=currency) currency=currency)
else: else:
@@ -611,23 +600,17 @@ class Item(LoggedModel):
invoice_address=invoice_address, invoice_address=invoice_address,
base_price_is='gross', base_price_is='gross',
currency=currency) currency=currency)
bundled_sum += bprice.gross if not self.tax_rule:
bundled_sum_net += bprice.net compare_price = TaxedPrice(gross=b.designated_price * b.count, net=b.designated_price * b.count,
bundled_sum_tax += bprice.tax tax=Decimal('0.00'), rate=Decimal('0.00'), name='')
else:
if not self.tax_rule: compare_price = self.tax_rule.tax(b.designated_price * b.count,
t = TaxedPrice(gross=price - bundled_sum, net=price - bundled_sum, tax=Decimal('0.00'), override_tax_rate=override_tax_rate,
rate=Decimal('0.00'), name='') invoice_address=invoice_address,
else: currency=currency)
t = self.tax_rule.tax(price, base_price_is=base_price_is, invoice_address=invoice_address, t.net += bprice.net - compare_price.net
override_tax_rate=override_tax_rate, currency=currency or self.event.currency, t.tax += bprice.tax - compare_price.tax
subtract_from_gross=bundled_sum) t.name = "MIXED!"
if bundled_sum:
t.name = "MIXED!"
t.gross += bundled_sum
t.net += bundled_sum_net
t.tax += bundled_sum_tax
return t return t
@@ -1022,16 +1005,6 @@ class ItemVariation(models.Model):
return False return False
return True return True
@property
def meta_data(self):
data = self.item.meta_data
if hasattr(self, 'meta_values_cached'):
data.update({v.property.name: v.value for v in self.meta_values_cached})
else:
data.update({v.property.name: v.value for v in self.meta_values.select_related('property').all()})
return OrderedDict((k, v) for k, v in sorted(data.items(), key=lambda k: k[0]))
class ItemAddOn(models.Model): class ItemAddOn(models.Model):
""" """
@@ -1408,10 +1381,8 @@ class Question(LoggedModel):
if self.type == Question.TYPE_CHOICE: if self.type == Question.TYPE_CHOICE:
if isinstance(answer, QuestionOption): if isinstance(answer, QuestionOption):
return answer return answer
if not isinstance(answer, (int, str)):
raise ValidationError(_('Invalid input type.'))
q = Q(identifier=answer) q = Q(identifier=answer)
if isinstance(answer, int) or (isinstance(answer, str) and answer.isdigit()): if isinstance(answer, int) or answer.isdigit():
q |= Q(pk=answer) q |= Q(pk=answer)
o = self.options.filter(q).first() o = self.options.filter(q).first()
if not o: if not o:
@@ -1811,21 +1782,8 @@ class ItemMetaValue(LoggedModel):
class Meta: class Meta:
unique_together = ('item', 'property') unique_together = ('item', 'property')
def delete(self, *args, **kwargs):
super().delete(*args, **kwargs)
class ItemVariationMetaValue(LoggedModel): def save(self, *args, **kwargs):
""" super().save(*args, **kwargs)
A meta-data value assigned to an item variation, overriding the value on the item.
:param variation: The variation this metadata is valid for
:type variation: ItemVariation
:param property: The property this value belongs to
:type property: ItemMetaProperty
:param value: The actual value
:type value: str
"""
variation = models.ForeignKey('ItemVariation', on_delete=models.CASCADE, related_name='meta_values')
property = models.ForeignKey('ItemMetaProperty', on_delete=models.CASCADE, related_name='variation_values')
value = models.TextField()
class Meta:
unique_together = ('variation', 'property')

View File

@@ -79,9 +79,7 @@ from pretix.base.services.locking import LOCK_TIMEOUT, NoLockManager
from pretix.base.settings import PERSON_NAME_SCHEMES from pretix.base.settings import PERSON_NAME_SCHEMES
from pretix.base.signals import order_gracefully_delete from pretix.base.signals import order_gracefully_delete
from ...helpers import OF_SELF
from ...helpers.countries import CachedCountries, FastCountryField from ...helpers.countries import CachedCountries, FastCountryField
from ...helpers.format import format_map
from ._transactions import ( from ._transactions import (
_fail, _transactions_mark_order_clean, _transactions_mark_order_dirty, _fail, _transactions_mark_order_clean, _transactions_mark_order_dirty,
) )
@@ -566,30 +564,17 @@ class Order(LockModel, LoggedModel):
@cached_property @cached_property
def user_cancel_fee(self): def user_cancel_fee(self):
fee = Decimal('0.00') fee = Decimal('0.00')
if self.status == Order.STATUS_PAID: if self.event.settings.cancel_allow_user_paid_keep_fees:
if self.event.settings.cancel_allow_user_paid_keep_fees: fee += self.fees.filter(
fee += self.fees.filter( fee_type__in=(OrderFee.FEE_TYPE_PAYMENT, OrderFee.FEE_TYPE_SHIPPING, OrderFee.FEE_TYPE_SERVICE,
fee_type__in=(OrderFee.FEE_TYPE_PAYMENT, OrderFee.FEE_TYPE_SHIPPING, OrderFee.FEE_TYPE_SERVICE, OrderFee.FEE_TYPE_CANCELLATION)
OrderFee.FEE_TYPE_CANCELLATION) ).aggregate(
).aggregate( s=Sum('value')
s=Sum('value') )['s'] or 0
)['s'] or 0 if self.event.settings.cancel_allow_user_paid_keep_percentage:
if self.event.settings.cancel_allow_user_paid_keep_percentage: fee += self.event.settings.cancel_allow_user_paid_keep_percentage / Decimal('100.0') * (self.total - fee)
fee += self.event.settings.cancel_allow_user_paid_keep_percentage / Decimal('100.0') * (self.total - fee) if self.event.settings.cancel_allow_user_paid_keep:
if self.event.settings.cancel_allow_user_paid_keep: fee += self.event.settings.cancel_allow_user_paid_keep
fee += self.event.settings.cancel_allow_user_paid_keep
else:
if self.event.settings.cancel_allow_user_unpaid_keep_fees:
fee += self.fees.filter(
fee_type__in=(OrderFee.FEE_TYPE_PAYMENT, OrderFee.FEE_TYPE_SHIPPING, OrderFee.FEE_TYPE_SERVICE,
OrderFee.FEE_TYPE_CANCELLATION)
).aggregate(
s=Sum('value')
)['s'] or 0
if self.event.settings.cancel_allow_user_unpaid_keep_percentage:
fee += self.event.settings.cancel_allow_user_unpaid_keep_percentage / Decimal('100.0') * (self.total - fee)
if self.event.settings.cancel_allow_user_unpaid_keep:
fee += self.event.settings.cancel_allow_user_unpaid_keep
return round_decimal(min(fee, self.total), self.event.currency) return round_decimal(min(fee, self.total), self.event.currency)
@property @property
@@ -657,12 +642,10 @@ class Order(LockModel, LoggedModel):
if self.user_cancel_deadline and now() > self.user_cancel_deadline: if self.user_cancel_deadline and now() > self.user_cancel_deadline:
return False return False
if self.status == Order.STATUS_PAID: if self.status == Order.STATUS_PAID or self.payment_refund_sum > Decimal('0.00'):
if self.total == Decimal('0.00'): if self.total == Decimal('0.00'):
return self.event.settings.cancel_allow_user return self.event.settings.cancel_allow_user
return self.event.settings.cancel_allow_user_paid return self.event.settings.cancel_allow_user_paid
elif self.payment_refund_sum > Decimal('0.00'):
return False
elif self.status == Order.STATUS_PENDING: elif self.status == Order.STATUS_PENDING:
return self.event.settings.cancel_allow_user return self.event.settings.cancel_allow_user
return False return False
@@ -809,7 +792,7 @@ class Order(LockModel, LoggedModel):
return True return True
ask_names = self.event.settings.get('attendee_names_asked', as_type=bool) ask_names = self.event.settings.get('attendee_names_asked', as_type=bool)
for cp in positions: for cp in positions:
if (cp.item.ask_attendee_data and ask_names) or cp.item.questions.all(): if (cp.item.admission and ask_names) or cp.item.questions.all():
return True return True
return False # nothing there to modify return False # nothing there to modify
@@ -998,7 +981,7 @@ class Order(LockModel, LoggedModel):
position and the attendee email will be used if available. position and the attendee email will be used if available.
""" """
from pretix.base.services.mail import ( from pretix.base.services.mail import (
SendMailException, mail, render_mail, SendMailException, TolerantDict, mail, render_mail,
) )
if not self.email and not (position and position.attendee_email): if not self.email and not (position and position.attendee_email):
@@ -1014,7 +997,7 @@ class Order(LockModel, LoggedModel):
try: try:
email_content = render_mail(template, context) email_content = render_mail(template, context)
subject = format_map(subject, context) subject = str(subject).format_map(TolerantDict(context))
mail( mail(
recipient, subject, template, context, recipient, subject, template, context,
self.event, self.locale, self, headers=headers, sender=sender, self.event, self.locale, self, headers=headers, sender=sender,
@@ -1044,7 +1027,7 @@ class Order(LockModel, LoggedModel):
with language(self.locale, self.event.settings.region): with language(self.locale, self.event.settings.region):
email_template = self.event.settings.mail_text_resend_link email_template = self.event.settings.mail_text_resend_link
email_context = get_email_context(event=self.event, order=self) email_context = get_email_context(event=self.event, order=self)
email_subject = self.event.settings.mail_subject_resend_link email_subject = _('Your order: %(code)s') % {'code': self.code}
self.send_mail( self.send_mail(
email_subject, email_template, email_context, email_subject, email_template, email_context,
'pretix.event.order.email.resend', user=user, auth=auth, 'pretix.event.order.email.resend', user=user, auth=auth,
@@ -1058,10 +1041,13 @@ class Order(LockModel, LoggedModel):
continue continue
yield op yield op
def create_transactions(self, is_new=False, positions=None, fees=None, dt_now=None, migrated=False, def create_transactions(self, *, source=None, is_new=False, positions=None, fees=None,
_backfill_before_cancellation=False, save=True): dt_now=None, migrated=False, _backfill_before_cancellation=False, save=True):
dt_now = dt_now or now() dt_now = dt_now or now()
if source is not None and (not isinstance(source, tuple) or len(source) != 2 or not all(isinstance(a, str) or a is None for a in source)):
return ValueError("source needs to be a 2-tuple of (source_type(str), source_identifier(str))")
# Count the transactions we already have # Count the transactions we already have
current_transaction_count = Counter() current_transaction_count = Counter()
if not is_new: if not is_new:
@@ -1106,6 +1092,8 @@ class Order(LockModel, LoggedModel):
tax_value=taxvalue, tax_value=taxvalue,
fee_type=feetype, fee_type=feetype,
internal_type=internaltype, internal_type=internaltype,
source_type=source[0] if source else None,
source_identifier=source[1] if source else None,
)) ))
create.sort(key=lambda t: (0 if t.count < 0 else 1, t.positionid or 0)) create.sort(key=lambda t: (0 if t.count < 0 else 1, t.positionid or 0))
if save: if save:
@@ -1511,9 +1499,6 @@ class OrderPayment(models.Model):
:type info: str :type info: str
:param fee: The ``OrderFee`` object used to track the fee for this order. :param fee: The ``OrderFee`` object used to track the fee for this order.
:type fee: pretix.base.models.OrderFee :type fee: pretix.base.models.OrderFee
:param process_initiated: Only for internal use inside pretix.presale to check which payments have started
the execution process.
:type process_initiated: bool
""" """
PAYMENT_STATE_CREATED = 'created' PAYMENT_STATE_CREATED = 'created'
PAYMENT_STATE_PENDING = 'pending' PAYMENT_STATE_PENDING = 'pending'
@@ -1564,9 +1549,6 @@ class OrderPayment(models.Model):
null=True, blank=True, related_name='payments', on_delete=models.SET_NULL null=True, blank=True, related_name='payments', on_delete=models.SET_NULL
) )
migrated = models.BooleanField(default=False) migrated = models.BooleanField(default=False)
process_initiated = models.BooleanField(
null=True # null = created before this field was introduced
)
objects = ScopedManager(organizer='order__event__organizer') objects = ScopedManager(organizer='order__event__organizer')
@@ -1596,7 +1578,7 @@ class OrderPayment(models.Model):
return self.order.event.get_payment_providers(cached=True).get(self.provider) return self.order.event.get_payment_providers(cached=True).get(self.provider)
@transaction.atomic() @transaction.atomic()
def _mark_paid_inner(self, force, count_waitinglist, user, auth, ignore_date=False, overpaid=False): def _mark_paid_inner(self, force, count_waitinglist, user, auth, ignore_date=False, overpaid=False, source=None):
from pretix.base.signals import order_paid from pretix.base.signals import order_paid
can_be_paid = self.order._can_be_paid(count_waitinglist=count_waitinglist, ignore_date=ignore_date, force=force) can_be_paid = self.order._can_be_paid(count_waitinglist=count_waitinglist, ignore_date=ignore_date, force=force)
if can_be_paid is not True: if can_be_paid is not True:
@@ -1619,7 +1601,9 @@ class OrderPayment(models.Model):
self.order.log_action('pretix.event.order.overpaid', {}, user=user, auth=auth) self.order.log_action('pretix.event.order.overpaid', {}, user=user, auth=auth)
order_paid.send(self.order.event, order=self.order) order_paid.send(self.order.event, order=self.order)
if status_change: if status_change:
self.order.create_transactions() self.order.create_transactions(
source=source or ('pretix.payment', None),
)
def fail(self, info=None, user=None, auth=None, log_data=None): def fail(self, info=None, user=None, auth=None, log_data=None):
""" """
@@ -1629,7 +1613,7 @@ class OrderPayment(models.Model):
been marked as paid. been marked as paid.
""" """
with transaction.atomic(): with transaction.atomic():
locked_instance = OrderPayment.objects.select_for_update(of=OF_SELF).get(pk=self.pk) locked_instance = OrderPayment.objects.select_for_update().get(pk=self.pk)
if locked_instance.state not in (OrderPayment.PAYMENT_STATE_CREATED, OrderPayment.PAYMENT_STATE_PENDING): if locked_instance.state not in (OrderPayment.PAYMENT_STATE_CREATED, OrderPayment.PAYMENT_STATE_PENDING):
# Race condition detected, this payment is already confirmed # Race condition detected, this payment is already confirmed
logger.info('Failed payment {} but ignored due to likely race condition.'.format( logger.info('Failed payment {} but ignored due to likely race condition.'.format(
@@ -1653,7 +1637,7 @@ class OrderPayment(models.Model):
}, user=user, auth=auth) }, user=user, auth=auth)
def confirm(self, count_waitinglist=True, send_mail=True, force=False, user=None, auth=None, mail_text='', def confirm(self, count_waitinglist=True, send_mail=True, force=False, user=None, auth=None, mail_text='',
ignore_date=False, lock=True, payment_date=None, generate_invoice=True): ignore_date=False, lock=True, payment_date=None, source=None):
""" """
Marks the payment as complete. If possible, this also marks the order as paid if no further Marks the payment as complete. If possible, this also marks the order as paid if no further
payment is required payment is required
@@ -1674,7 +1658,7 @@ class OrderPayment(models.Model):
:raises Quota.QuotaExceededException: if the quota is exceeded and ``force`` is ``False`` :raises Quota.QuotaExceededException: if the quota is exceeded and ``force`` is ``False``
""" """
with transaction.atomic(): with transaction.atomic():
locked_instance = OrderPayment.objects.select_for_update(of=OF_SELF).get(pk=self.pk) locked_instance = OrderPayment.objects.select_for_update().get(pk=self.pk)
if locked_instance.state == self.PAYMENT_STATE_CONFIRMED: if locked_instance.state == self.PAYMENT_STATE_CONFIRMED:
# Race condition detected, this payment is already confirmed # Race condition detected, this payment is already confirmed
logger.info('Confirmed payment {} but ignored due to likely race condition.'.format( logger.info('Confirmed payment {} but ignored due to likely race condition.'.format(
@@ -1717,10 +1701,10 @@ class OrderPayment(models.Model):
return return
self._mark_order_paid(count_waitinglist, send_mail, force, user, auth, mail_text, ignore_date, lock, payment_sum - refund_sum, self._mark_order_paid(count_waitinglist, send_mail, force, user, auth, mail_text, ignore_date, lock, payment_sum - refund_sum,
generate_invoice) source)
def _mark_order_paid(self, count_waitinglist=True, send_mail=True, force=False, user=None, auth=None, mail_text='', def _mark_order_paid(self, count_waitinglist=True, send_mail=True, force=False, user=None, auth=None, mail_text='',
ignore_date=False, lock=True, payment_refund_sum=0, allow_generate_invoice=True): ignore_date=False, lock=True, payment_refund_sum=0, source=None):
from pretix.base.services.invoices import ( from pretix.base.services.invoices import (
generate_invoice, invoice_qualified, generate_invoice, invoice_qualified,
) )
@@ -1734,10 +1718,10 @@ class OrderPayment(models.Model):
with lockfn(): with lockfn():
self._mark_paid_inner(force, count_waitinglist, user, auth, overpaid=payment_refund_sum > self.order.total, self._mark_paid_inner(force, count_waitinglist, user, auth, overpaid=payment_refund_sum > self.order.total,
ignore_date=ignore_date) ignore_date=ignore_date, source=source)
invoice = None invoice = None
if invoice_qualified(self.order) and allow_generate_invoice: if invoice_qualified(self.order):
invoices = self.order.invoices.filter(is_cancellation=False).count() invoices = self.order.invoices.filter(is_cancellation=False).count()
cancellations = self.order.invoices.filter(is_cancellation=True).count() cancellations = self.order.invoices.filter(is_cancellation=True).count()
gen_invoice = ( gen_invoice = (
@@ -1762,8 +1746,8 @@ class OrderPayment(models.Model):
with language(self.order.locale, self.order.event.settings.region): with language(self.order.locale, self.order.event.settings.region):
email_template = self.order.event.settings.mail_text_order_paid_attendee email_template = self.order.event.settings.mail_text_order_paid_attendee
email_subject = self.order.event.settings.mail_subject_order_paid_attendee
email_context = get_email_context(event=self.order.event, order=self.order, position=position) email_context = get_email_context(event=self.order.event, order=self.order, position=position)
email_subject = _('Event registration confirmed: %(code)s') % {'code': self.order.code}
try: try:
position.send_mail( position.send_mail(
email_subject, email_template, email_context, email_subject, email_template, email_context,
@@ -1780,8 +1764,8 @@ class OrderPayment(models.Model):
with language(self.order.locale, self.order.event.settings.region): with language(self.order.locale, self.order.event.settings.region):
email_template = self.order.event.settings.mail_text_order_paid email_template = self.order.event.settings.mail_text_order_paid
email_subject = self.order.event.settings.mail_subject_order_paid
email_context = get_email_context(event=self.order.event, order=self.order, payment_info=mail_text) email_context = get_email_context(event=self.order.event, order=self.order, payment_info=mail_text)
email_subject = _('Payment received for your order: %(code)s') % {'code': self.order.code}
try: try:
self.order.send_mail( self.order.send_mail(
email_subject, email_template, email_context, email_subject, email_template, email_context,
@@ -2416,7 +2400,7 @@ class OrderPosition(AbstractPosition):
:param attach_ical: Attach relevant ICS files :param attach_ical: Attach relevant ICS files
""" """
from pretix.base.services.mail import ( from pretix.base.services.mail import (
SendMailException, mail, render_mail, SendMailException, TolerantDict, mail, render_mail,
) )
if not self.attendee_email: if not self.attendee_email:
@@ -2429,7 +2413,7 @@ class OrderPosition(AbstractPosition):
recipient = self.attendee_email recipient = self.attendee_email
try: try:
email_content = render_mail(template, context) email_content = render_mail(template, context)
subject = format_map(subject, context) subject = str(subject).format_map(TolerantDict(context))
mail( mail(
recipient, subject, template, context, recipient, subject, template, context,
self.event, self.order.locale, order=self.order, headers=headers, sender=sender, self.event, self.order.locale, order=self.order, headers=headers, sender=sender,
@@ -2461,7 +2445,7 @@ class OrderPosition(AbstractPosition):
with language(self.order.locale, self.order.event.settings.region): with language(self.order.locale, self.order.event.settings.region):
email_template = self.event.settings.mail_text_resend_link email_template = self.event.settings.mail_text_resend_link
email_context = get_email_context(event=self.order.event, order=self.order, position=self) email_context = get_email_context(event=self.order.event, order=self.order, position=self)
email_subject = self.event.settings.mail_subject_resend_link email_subject = _('Your event registration: %(code)s') % {'code': self.order.code}
self.send_mail( self.send_mail(
email_subject, email_template, email_context, email_subject, email_template, email_context,
'pretix.event.order.email.resend', user=user, auth=auth, 'pretix.event.order.email.resend', user=user, auth=auth,
@@ -2507,6 +2491,8 @@ class Transaction(models.Model):
:param id: ID of the transaction :param id: ID of the transaction
:param order: Order the transaction belongs to :param order: Order the transaction belongs to
:param source_type: Functionality that caused the transaction to be created, usually the name of a module or plugin
:param source_identifier: Identifier of the entity that caused the transaction to be created, as defined by the module or plugin noted in ``source_type``.
:param datetime: Date and time of the transaction :param datetime: Date and time of the transaction
:param migrated: Whether this object was reconstructed because the order was created before transactions where introduced :param migrated: Whether this object was reconstructed because the order was created before transactions where introduced
:param positionid: Affected Position ID, in case this transaction represents a change in an order position :param positionid: Affected Position ID, in case this transaction represents a change in an order position
@@ -2529,6 +2515,12 @@ class Transaction(models.Model):
related_name='transactions', related_name='transactions',
on_delete=models.PROTECT on_delete=models.PROTECT
) )
source_type = models.CharField(
max_length=190, db_index=True, null=True, blank=True
)
source_identifier = models.CharField(
max_length=190, db_index=True, null=True, blank=True
)
created = models.DateTimeField( created = models.DateTimeField(
auto_now_add=True, auto_now_add=True,
db_index=True, db_index=True,
@@ -2689,7 +2681,7 @@ class CartPosition(AbstractPosition):
category_key = (self.item.category.position, self.item.category.id) if self.item.category_id is not None else (0, 0) category_key = (self.item.category.position, self.item.category.id) if self.item.category_id is not None else (0, 0)
item_key = self.item.position, self.item_id item_key = self.item.position, self.item_id
variation_key = (self.variation.position, self.variation.id) if self.variation_id is not None else (0, 0) variation_key = (self.variation.position, self.variation.id) if self.variation_id is not None else (0, 0)
line_key = (self.price, (self.voucher_id or 0), (self.seat.sorting_rank if self.seat_id else 0), self.pk) line_key = (self.price, (self.voucher_id or 0), (self.seat.sorting_rank if self.seat_id else None), self.pk)
sort_key = subevent_key + category_key + item_key + variation_key + line_key sort_key = subevent_key + category_key + item_key + variation_key + line_key
if self.addon_to_id: if self.addon_to_id:
@@ -2746,7 +2738,6 @@ class CartPosition(AbstractPosition):
tax_rule=self.item.tax_rule, tax_rule=self.item.tax_rule,
invoice_address=invoice_address, invoice_address=invoice_address,
bundled_sum=sum([b.price_after_voucher for b in bundled_positions]), bundled_sum=sum([b.price_after_voucher for b in bundled_positions]),
is_bundled=self.is_bundled,
) )
if line_price.gross != self.line_price_gross or line_price.rate != self.tax_rate: if line_price.gross != self.line_price_gross or line_price.rate != self.tax_rate:
self.line_price_gross = line_price.gross self.line_price_gross = line_price.gross

View File

@@ -23,7 +23,6 @@ import json
from decimal import Decimal from decimal import Decimal
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.core.validators import MaxValueValidator, MinValueValidator
from django.db import models from django.db import models
from django.utils.formats import localize from django.utils.formats import localize
from django.utils.translation import gettext_lazy as _, pgettext from django.utils.translation import gettext_lazy as _, pgettext
@@ -150,15 +149,7 @@ class TaxRule(LoggedModel):
rate = models.DecimalField( rate = models.DecimalField(
max_digits=10, max_digits=10,
decimal_places=2, decimal_places=2,
validators=[ verbose_name=_("Tax rate")
MaxValueValidator(
limit_value=Decimal("100.00"),
),
MinValueValidator(
limit_value=Decimal("0.00"),
),
],
verbose_name=_("Tax rate"),
) )
price_includes_tax = models.BooleanField( price_includes_tax = models.BooleanField(
verbose_name=_("The configured product prices include the tax amount"), verbose_name=_("The configured product prices include the tax amount"),

View File

@@ -137,8 +137,6 @@ class Voucher(LoggedModel):
:type max_usages: int :type max_usages: int
:param redeemed: The number of times this voucher already has been redeemed :param redeemed: The number of times this voucher already has been redeemed
:type redeemed: int :type redeemed: int
:param min_usages: The minimum number of times this voucher must be redeemed
:type min_usages: int
:param valid_until: The expiration date of this voucher (optional) :param valid_until: The expiration date of this voucher (optional)
:type valid_until: datetime :type valid_until: datetime
:param block_quota: If set to true, this voucher will reserve quota for its holder :param block_quota: If set to true, this voucher will reserve quota for its holder
@@ -201,14 +199,6 @@ class Voucher(LoggedModel):
verbose_name=_("Redeemed"), verbose_name=_("Redeemed"),
default=0 default=0
) )
min_usages = models.PositiveIntegerField(
verbose_name=_("Minimum usages"),
help_text=_("If set to more than one, the voucher must be redeemed for this many products when it is used for "
"the first time. On later usages, it can also be used for lower numbers of products. Note that "
"this means that the total number of usages in some cases can be lower than this limit, e.g. in "
"case of cancellations."),
default=1
)
budget = models.DecimalField( budget = models.DecimalField(
verbose_name=_("Maximum discount budget"), verbose_name=_("Maximum discount budget"),
help_text=_("This is the maximum monetary amount that will be discounted using this voucher across all usages. " help_text=_("This is the maximum monetary amount that will be discounted using this voucher across all usages. "
@@ -360,10 +350,6 @@ class Voucher(LoggedModel):
'redeemed': redeemed 'redeemed': redeemed
} }
) )
if data.get('max_usages', 1) < data.get('min_usages', 1):
raise ValidationError(
_('The maximum number of usages may not be lower than the minimum number of usages.'),
)
@staticmethod @staticmethod
def clean_subevent(data, event): def clean_subevent(data, event):
@@ -478,7 +464,7 @@ class Voucher(LoggedModel):
if quota: if quota:
raise ValidationError(_('You need to choose a specific product if you select a seat.')) raise ValidationError(_('You need to choose a specific product if you select a seat.'))
if data.get('max_usages', 1) > 1 or data.get('min_usages', 1) > 1: if data.get('max_usages', 1) > 1:
raise ValidationError(_('Seat-specific vouchers can only be used once.')) raise ValidationError(_('Seat-specific vouchers can only be used once.'))
if item and seat.product != item: if item and seat.product != item:
@@ -581,10 +567,6 @@ class Voucher(LoggedModel):
else: else:
return bool(subevent.seating_plan) if subevent else self.event.seating_plan return bool(subevent.seating_plan) if subevent else self.event.seating_plan
@property
def min_usages_remaining(self):
return max(1, self.min_usages - self.redeemed)
@classmethod @classmethod
def annotate_budget_used_orders(cls, qs): def annotate_budget_used_orders(cls, qs):
opq = OrderPosition.objects.filter( opq = OrderPosition.objects.filter(

View File

@@ -20,7 +20,6 @@
# <https://www.gnu.org/licenses/>. # <https://www.gnu.org/licenses/>.
# #
from datetime import timedelta from datetime import timedelta
from typing import Any, Dict, Union
from django.core.exceptions import ObjectDoesNotExist, ValidationError from django.core.exceptions import ObjectDoesNotExist, ValidationError
from django.db import models, transaction from django.db import models, transaction
@@ -28,16 +27,14 @@ from django.db.models import F, Q, Sum
from django.utils.timezone import now from django.utils.timezone import now
from django.utils.translation import gettext_lazy as _, pgettext_lazy from django.utils.translation import gettext_lazy as _, pgettext_lazy
from django_scopes import ScopedManager from django_scopes import ScopedManager
from i18nfield.strings import LazyI18nString
from phonenumber_field.modelfields import PhoneNumberField from phonenumber_field.modelfields import PhoneNumberField
from pretix.base.email import get_email_context from pretix.base.email import get_email_context
from pretix.base.i18n import language from pretix.base.i18n import language
from pretix.base.models import User, Voucher from pretix.base.models import Voucher
from pretix.base.services.mail import SendMailException, mail, render_mail from pretix.base.services.mail import mail
from pretix.base.settings import PERSON_NAME_SCHEMES from pretix.base.settings import PERSON_NAME_SCHEMES
from ...helpers.format import format_map
from .base import LoggedModel from .base import LoggedModel
from .event import Event, SubEvent from .event import Event, SubEvent
from .items import Item, ItemVariation from .items import Item, ItemVariation
@@ -216,74 +213,15 @@ class WaitingListEntry(LoggedModel):
self.voucher = v self.voucher = v
self.save() self.save()
self.send_mail(
self.event.settings.mail_subject_waiting_list,
self.event.settings.mail_text_waiting_list,
get_email_context(
event=self.event,
waiting_list_entry=self,
waiting_list_voucher=v,
event_or_subevent=self.subevent or self.event,
),
user=user,
auth=auth,
)
def send_mail(self, subject: Union[str, LazyI18nString], template: Union[str, LazyI18nString],
context: Dict[str, Any]=None, log_entry_type: str='pretix.waitinglist.email.sent',
user: User=None, headers: dict=None, sender: str=None, auth=None, auto_email=True,
attach_other_files: list=None, attach_cached_files: list=None):
"""
Sends an email to the entry's contact address.
* Call ``pretix.base.services.mail.mail`` with useful values for the ``event``, ``locale``, and ``recipient``
parameters.
* Create a ``LogEntry`` with the email contents.
:param subject: Subject of the email
:param template: LazyI18nString or template filename, see ``pretix.base.services.mail.mail`` for more details
:param context: Dictionary to use for rendering the template
:param log_entry_type: Key to be used for the log entry
:param user: Administrative user who triggered this mail to be sent
:param headers: Dictionary with additional mail headers
:param sender: Custom email sender.
"""
if not self.email:
return
for k, v in self.event.meta_data.items():
context['meta_' + k] = v
with language(self.locale, self.event.settings.region): with language(self.locale, self.event.settings.region):
recipient = self.email mail(
self.email,
try: _('You have been selected from the waitinglist for {event}').format(event=str(self.event)),
email_content = render_mail(template, context) self.event.settings.mail_text_waiting_list,
subject = format_map(subject, context) get_email_context(event=self.event, waiting_list_entry=self),
mail( self.event,
recipient, subject, template, context, locale=self.locale
self.event, )
self.locale,
headers=headers,
sender=sender,
auto_email=auto_email,
attach_other_files=attach_other_files,
attach_cached_files=attach_cached_files,
)
except SendMailException:
raise
else:
self.log_action(
log_entry_type,
user=user,
auth=auth,
data={
'subject': subject,
'message': email_content,
'recipient': recipient,
}
)
@staticmethod @staticmethod
def clean_itemvar(event, item, variation): def clean_itemvar(event, item, variation):

View File

@@ -63,15 +63,14 @@ from pretix.base.models import (
OrderRefund, Quota, OrderRefund, Quota,
) )
from pretix.base.reldate import RelativeDateField, RelativeDateWrapper from pretix.base.reldate import RelativeDateField, RelativeDateWrapper
from pretix.base.services.cart import get_fees
from pretix.base.settings import SettingsSandbox from pretix.base.settings import SettingsSandbox
from pretix.base.signals import register_payment_providers from pretix.base.signals import register_payment_providers
from pretix.base.templatetags.money import money_filter from pretix.base.templatetags.money import money_filter
from pretix.base.templatetags.rich_text import rich_text from pretix.base.templatetags.rich_text import rich_text
from pretix.helpers import OF_SELF
from pretix.helpers.countries import CachedCountries from pretix.helpers.countries import CachedCountries
from pretix.helpers.format import format_map
from pretix.helpers.money import DecimalTextInput from pretix.helpers.money import DecimalTextInput
from pretix.multidomain.urlreverse import build_absolute_uri from pretix.multidomain.urlreverse import build_absolute_uri, eventreverse
from pretix.presale.views import get_cart, get_cart_total from pretix.presale.views import get_cart, get_cart_total
from pretix.presale.views.cart import cart_session, get_or_create_cart_id from pretix.presale.views.cart import cart_session, get_or_create_cart_id
@@ -139,50 +138,6 @@ class BasePaymentProvider:
""" """
return self.settings.get('_enabled', as_type=bool) return self.settings.get('_enabled', as_type=bool)
@property
def multi_use_supported(self) -> bool:
"""
Returns whether or whether not this payment provider supports being used multiple times in the same
checkout, or in addition to a different payment provider. This is usually only useful for payment providers
that represent gift cards, i.e. payment methods with an upper limit per payment instrument that can usually
be combined with other instruments.
If you set this property to ``True``, the behavior of how pretix interacts with your payment provider changes
and you will need to respect the following rules:
- ``payment_form_render`` must not depend on session state, it must always allow a user to add a new payment.
Editing a payment is not possible, but pretix will give users an option to delete it.
- Returning ``True`` from ``checkout_prepare`` is no longer enough. Instead, you must *also* call
``pretix.base.services.cart.add_payment_to_cart(request, provider, min_value, max_value, info_data)``
to add the payment to the session. You are still allowed to do a redirect from ``checkout_prepare`` and then
call this function upon return.
- Unlike in the general case, when ``checkout_prepare`` is called, the ``cart['total']`` parameter will _not yet_
include payment fees charged by your provider as we don't yet know the amount of the charge, so you need to
take care of that yourself when setting your maximum amount.
- ``payment_is_valid_session`` will not be called during checkout, don't rely on it. If you called
``add_payment_to_cart``, we'll trust the payment is okay and your next chance to change that will be
``execute_payment``.
The changed behavior currently only affects the behavior during initial checkout (i.e. ``checkout_prepare``),
for ``payment_prepare`` the regular behavior applies and you are expected to just modify the amount of the
``OrderPayment`` object if you need to.
"""
return False
@property
def execute_payment_needs_user(self) -> bool:
"""
Set this to ``True`` if your ``execute_payment`` function needs to be triggered by a user request, i.e. either
needs the ``request`` object or might require a browser redirect. If this is ``False``, you will not receive
a ``request`` and may not redirect since execute_payment might be called server-side. You should ensure that
your ``execute_payment`` method has a limited execution time (i.e. by using ``timeout`` for all external calls)
and handles all error cases appropriately.
"""
return True
@property @property
def test_mode_message(self) -> str: def test_mode_message(self) -> str:
""" """
@@ -326,6 +281,16 @@ class BasePaymentProvider:
help_text=_('Users will not be able to choose this payment provider after the given date.'), help_text=_('Users will not be able to choose this payment provider after the given date.'),
required=False, required=False,
)), )),
('_invoice_text',
I18nFormField(
label=_('Text on invoices'),
help_text=_('Will be printed just below the payment figures and above the closing text on invoices. '
'This will only be used if the invoice is generated before the order is paid. If the '
'invoice is generated later, it will show a text stating that it has already been paid.'),
required=False,
widget=I18nTextarea,
widget_kwargs={'attrs': {'rows': '2'}}
)),
('_total_min', ('_total_min',
forms.DecimalField( forms.DecimalField(
label=_('Minimum order total'), label=_('Minimum order total'),
@@ -373,16 +338,6 @@ class BasePaymentProvider:
'above!').format(docs_url='https://docs.pretix.eu/en/latest/user/payments/fees.html'), 'above!').format(docs_url='https://docs.pretix.eu/en/latest/user/payments/fees.html'),
required=False required=False
)), )),
('_invoice_text',
I18nFormField(
label=_('Text on invoices'),
help_text=_('Will be printed just below the payment figures and above the closing text on invoices. '
'This will only be used if the invoice is generated before the order is paid. If the '
'invoice is generated later, it will show a text stating that it has already been paid.'),
required=False,
widget=I18nTextarea,
widget_kwargs={'attrs': {'rows': '2'}}
)),
('_restricted_countries', ('_restricted_countries',
forms.MultipleChoiceField( forms.MultipleChoiceField(
label=_('Restrict to countries'), label=_('Restrict to countries'),
@@ -619,7 +574,7 @@ class BasePaymentProvider:
ctx = {'request': request, 'form': form} ctx = {'request': request, 'form': form}
return template.render(ctx) return template.render(ctx)
def checkout_confirm_render(self, request, order: Order=None, info_data: dict=None) -> str: def checkout_confirm_render(self, request, order: Order=None) -> str:
""" """
If the user has successfully filled in their payment data, they will be redirected If the user has successfully filled in their payment data, they will be redirected
to a confirmation page which lists all details of their order for a final review. to a confirmation page which lists all details of their order for a final review.
@@ -629,9 +584,7 @@ class BasePaymentProvider:
In most cases, this should include a short summary of the user's input and In most cases, this should include a short summary of the user's input and
a short explanation on how the payment process will continue. a short explanation on how the payment process will continue.
:param request: The current HTTP request.
:param order: Only set when this is a change to a new payment method for an existing order. :param order: Only set when this is a change to a new payment method for an existing order.
:param info_data: The ``info_data`` dictionary you set during ``add_payment_to_cart`` (only filled if ``multi_use_supported`` is set)
""" """
raise NotImplementedError() # NOQA raise NotImplementedError() # NOQA
@@ -665,10 +618,6 @@ class BasePaymentProvider:
.. IMPORTANT:: If this is called, the user has not yet confirmed their order. .. IMPORTANT:: If this is called, the user has not yet confirmed their order.
You may NOT do anything which actually moves money. You may NOT do anything which actually moves money.
Note: The behavior of this method changes significantly when you set
``multi_use_supported``. Please refer to the ``multi_use_supported`` documentation
for more information.
:param cart: This dictionary contains at least the following keys: :param cart: This dictionary contains at least the following keys:
positions: positions:
@@ -708,9 +657,9 @@ class BasePaymentProvider:
You will be passed an :py:class:`pretix.base.models.OrderPayment` object that contains You will be passed an :py:class:`pretix.base.models.OrderPayment` object that contains
the amount of money that should be paid. the amount of money that should be paid.
If you need any special behavior, you can return a string containing the URL the user will be redirected to. If you need any special behavior, you can return a string
If you are done with your process you should return the user to the order's detail page. Redirection is not containing the URL the user will be redirected to. If you are done with your process
allowed if you set ``execute_payment_needs_user`` to ``True``. you should return the user to the order's detail page.
If the payment is completed, you should call ``payment.confirm()``. Please note that this might If the payment is completed, you should call ``payment.confirm()``. Please note that this might
raise a ``Quota.QuotaExceededException`` if (and only if) the payment term of this order is over and raise a ``Quota.QuotaExceededException`` if (and only if) the payment term of this order is over and
@@ -722,7 +671,7 @@ class BasePaymentProvider:
On errors, you should raise a ``PaymentException``. On errors, you should raise a ``PaymentException``.
:param request: A HTTP request, except if ``execute_payment_needs_user`` is ``False`` :param order: The order object
:param payment: An ``OrderPayment`` instance :param payment: An ``OrderPayment`` instance
""" """
return None return None
@@ -928,15 +877,6 @@ class BasePaymentProvider:
""" """
return {} return {}
def api_refund_details(self, refund: OrderRefund):
"""
Will be called to populate the ``details`` parameter of the refund in the REST API.
:param refund: The refund in question.
:return: A serializable dictionary
"""
return {}
def matching_id(self, payment: OrderPayment): def matching_id(self, payment: OrderPayment):
""" """
Will be called to get an ID for matching this payment when comparing pretix records with records of an external Will be called to get an ID for matching this payment when comparing pretix records with records of an external
@@ -956,7 +896,6 @@ class FreeOrderProvider(BasePaymentProvider):
is_implicit = True is_implicit = True
is_enabled = True is_enabled = True
identifier = "free" identifier = "free"
execute_payment_needs_user = False
def checkout_confirm_render(self, request: HttpRequest) -> str: def checkout_confirm_render(self, request: HttpRequest) -> str:
return _("No payment is required as this order only includes products which are free of charge.") return _("No payment is required as this order only includes products which are free of charge.")
@@ -1020,9 +959,6 @@ class BoxOfficeProvider(BasePaymentProvider):
"payment_data": payment.info_data.get('payment_data', {}), "payment_data": payment.info_data.get('payment_data', {}),
} }
def api_refund_details(self, refund: OrderRefund):
return self.api_payment_details(refund)
def payment_control_render(self, request, payment) -> str: def payment_control_render(self, request, payment) -> str:
if not payment.info: if not payment.info:
return return
@@ -1043,7 +979,6 @@ class BoxOfficeProvider(BasePaymentProvider):
class ManualPayment(BasePaymentProvider): class ManualPayment(BasePaymentProvider):
identifier = 'manual' identifier = 'manual'
verbose_name = _('Manual payment') verbose_name = _('Manual payment')
execute_payment_needs_user = False
@property @property
def test_mode_message(self): def test_mode_message(self):
@@ -1124,12 +1059,12 @@ class ManualPayment(BasePaymentProvider):
} }
def order_pending_mail_render(self, order, payment) -> str: def order_pending_mail_render(self, order, payment) -> str:
msg = format_map(self.settings.get('email_instructions', as_type=LazyI18nString), self.format_map(order, payment)) msg = str(self.settings.get('email_instructions', as_type=LazyI18nString)).format_map(self.format_map(order, payment))
return msg return msg
def payment_pending_render(self, request, payment) -> str: def payment_pending_render(self, request, payment) -> str:
return rich_text( return rich_text(
format_map(self.settings.get('pending_description', as_type=LazyI18nString), self.format_map(payment.order, payment)) str(self.settings.get('pending_description', as_type=LazyI18nString)).format_map(self.format_map(payment.order, payment))
) )
@@ -1184,42 +1119,18 @@ class OffsettingProvider(BasePaymentProvider):
class GiftCardPayment(BasePaymentProvider): class GiftCardPayment(BasePaymentProvider):
identifier = "giftcard" identifier = "giftcard"
priority = 10
multi_use_supported = True
execute_payment_needs_user = False
verbose_name = _("Gift card") verbose_name = _("Gift card")
priority = 10
@property
def public_name(self) -> str:
return str(self.settings.get("public_name", as_type=LazyI18nString)) or _(
"Gift card"
)
@property @property
def settings_form_fields(self): def settings_form_fields(self):
fields = [ f = super().settings_form_fields
(
"public_name",
I18nFormField(
label=_("Payment method name"), widget=I18nTextInput, required=False
),
),
(
"public_description",
I18nFormField(
label=_("Payment method description"), widget=I18nTextarea, required=False
),
),
]
f = OrderedDict(fields + list(super().settings_form_fields.items()))
del f['_fee_abs'] del f['_fee_abs']
del f['_fee_percent'] del f['_fee_percent']
del f['_fee_reverse_calc'] del f['_fee_reverse_calc']
del f['_total_min'] del f['_total_min']
del f['_total_max'] del f['_total_max']
del f['_invoice_text'] del f['_invoice_text']
f.move_to_end("_enabled", last=False)
return f return f
@property @property
@@ -1233,14 +1144,10 @@ class GiftCardPayment(BasePaymentProvider):
return super().order_change_allowed(order) and self.event.organizer.has_gift_cards return super().order_change_allowed(order) and self.event.organizer.has_gift_cards
def payment_form_render(self, request: HttpRequest, total: Decimal) -> str: def payment_form_render(self, request: HttpRequest, total: Decimal) -> str:
return get_template('pretixcontrol/giftcards/checkout.html').render({ return get_template('pretixcontrol/giftcards/checkout.html').render({})
'request': request,
})
def checkout_confirm_render(self, request, order=None, info_data=None) -> str: def checkout_confirm_render(self, request) -> str:
return get_template('pretixcontrol/giftcards/checkout_confirm.html').render({ return get_template('pretixcontrol/giftcards/checkout_confirm.html').render({})
'info_data': info_data,
})
def refund_control_render(self, request, refund) -> str: def refund_control_render(self, request, refund) -> str:
from .models import GiftCard from .models import GiftCard
@@ -1284,9 +1191,6 @@ class GiftCardPayment(BasePaymentProvider):
} }
} }
def api_refund_details(self, refund: OrderRefund):
return self.api_payment_details(refund)
def payment_partial_refund_supported(self, payment: OrderPayment) -> bool: def payment_partial_refund_supported(self, payment: OrderPayment) -> bool:
return True return True
@@ -1294,8 +1198,6 @@ class GiftCardPayment(BasePaymentProvider):
return True return True
def checkout_prepare(self, request: HttpRequest, cart: Dict[str, Any]) -> Union[bool, str, None]: def checkout_prepare(self, request: HttpRequest, cart: Dict[str, Any]) -> Union[bool, str, None]:
from pretix.base.services.cart import add_payment_to_cart
for p in get_cart(request): for p in get_cart(request):
if p.item.issue_giftcard: if p.item.issue_giftcard:
messages.error(request, _("You cannot pay with gift cards when buying a gift card.")) messages.error(request, _("You cannot pay with gift cards when buying a gift card."))
@@ -1304,7 +1206,7 @@ class GiftCardPayment(BasePaymentProvider):
cs = cart_session(request) cs = cart_session(request)
try: try:
gc = self.event.organizer.accepted_gift_cards.get( gc = self.event.organizer.accepted_gift_cards.get(
secret=request.POST.get("giftcard").strip() secret=request.POST.get("giftcard")
) )
if gc.currency != self.event.currency: if gc.currency != self.event.currency:
messages.error(request, _("This gift card does not support this currency.")) messages.error(request, _("This gift card does not support this currency."))
@@ -1321,22 +1223,34 @@ class GiftCardPayment(BasePaymentProvider):
if gc.value <= Decimal("0.00"): if gc.value <= Decimal("0.00"):
messages.error(request, _("All credit on this gift card has been used.")) messages.error(request, _("All credit on this gift card has been used."))
return return
if 'gift_cards' not in cs:
cs['gift_cards'] = []
elif gc.pk in cs['gift_cards']:
messages.error(request, _("This gift card is already used for your payment."))
return
cs['gift_cards'] = cs['gift_cards'] + [gc.pk]
for p in cs.get('payments', []): total = sum(p.total for p in cart['positions'])
if p['provider'] == self.identifier and p['info_data']['gift_card'] == gc.pk: # Recompute fees. Some plugins, e.g. pretix-servicefees, change their fee schedule if a gift card is
messages.error(request, _("This gift card is already used for your payment.")) # applied.
return fees = get_fees(
self.event, request, total, cart['invoice_address'], cs.get('payment'),
add_payment_to_cart( cart['raw']
request,
self,
max_value=gc.value,
info_data={
'gift_card': gc.pk,
'gift_card_secret': gc.secret,
}
) )
return True total += sum([f.value for f in fees])
remainder = total
if remainder > Decimal('0.00'):
del cs['payment']
messages.success(request, _("Your gift card has been applied, but {} still need to be paid. Please select a payment method.").format(
money_filter(remainder, self.event.currency)
))
else:
messages.success(request, _("Your gift card has been applied."))
kwargs = {'step': 'payment'}
if request.resolver_match and 'cart_namespace' in request.resolver_match.kwargs:
kwargs['cart_namespace'] = request.resolver_match.kwargs['cart_namespace']
return eventreverse(self.event, 'presale:event.checkout', kwargs=kwargs)
except GiftCard.DoesNotExist: except GiftCard.DoesNotExist:
if self.event.vouchers.filter(code__iexact=request.POST.get("giftcard")).exists(): if self.event.vouchers.filter(code__iexact=request.POST.get("giftcard")).exists():
messages.warning(request, _("You entered a voucher instead of a gift card. Vouchers can only be entered on the first page of the shop below " messages.warning(request, _("You entered a voucher instead of a gift card. Vouchers can only be entered on the first page of the shop below "
@@ -1354,7 +1268,7 @@ class GiftCardPayment(BasePaymentProvider):
try: try:
gc = self.event.organizer.accepted_gift_cards.get( gc = self.event.organizer.accepted_gift_cards.get(
secret=request.POST.get("giftcard").strip() secret=request.POST.get("giftcard")
) )
if gc.currency != self.event.currency: if gc.currency != self.event.currency:
messages.error(request, _("This gift card does not support this currency.")) messages.error(request, _("This gift card does not support this currency."))
@@ -1373,7 +1287,6 @@ class GiftCardPayment(BasePaymentProvider):
return return
payment.info_data = { payment.info_data = {
'gift_card': gc.pk, 'gift_card': gc.pk,
'gift_card_secret': gc.secret,
'retry': True 'retry': True
} }
payment.amount = min(payment.amount, gc.value) payment.amount = min(payment.amount, gc.value)
@@ -1381,7 +1294,7 @@ class GiftCardPayment(BasePaymentProvider):
return True return True
except GiftCard.DoesNotExist: except GiftCard.DoesNotExist:
if self.event.vouchers.filter(code__iexact=request.POST.get("giftcard").strip()).exists(): if self.event.vouchers.filter(code__iexact=request.POST.get("giftcard")).exists():
messages.warning(request, _("You entered a voucher instead of a gift card. Vouchers can only be entered on the first page of the shop below " messages.warning(request, _("You entered a voucher instead of a gift card. Vouchers can only be entered on the first page of the shop below "
"the product selection.")) "the product selection."))
else: else:
@@ -1389,46 +1302,37 @@ class GiftCardPayment(BasePaymentProvider):
except GiftCard.MultipleObjectsReturned: except GiftCard.MultipleObjectsReturned:
messages.error(request, _("This gift card can not be redeemed since its code is not unique. Please contact the organizer of this event.")) messages.error(request, _("This gift card can not be redeemed since its code is not unique. Please contact the organizer of this event."))
def execute_payment(self, request: HttpRequest, payment: OrderPayment, is_early_special_case=False) -> str: def execute_payment(self, request: HttpRequest, payment: OrderPayment) -> str:
for p in payment.order.positions.all(): # This method will only be called when retrying payments, e.g. after a payment_prepare call. It is not called
# during the order creation phase because this payment provider is a special case.
for p in payment.order.positions.all(): # noqa - just a safeguard
if p.item.issue_giftcard: if p.item.issue_giftcard:
raise PaymentException(_("You cannot pay with gift cards when buying a gift card.")) raise PaymentException(_("You cannot pay with gift cards when buying a gift card."))
gcpk = payment.info_data.get('gift_card') gcpk = payment.info_data.get('gift_card')
if not gcpk: if not gcpk or not payment.info_data.get('retry'):
raise PaymentException("Invalid state, should never occur.") raise PaymentException("Invalid state, should never occur.")
try: with transaction.atomic():
with transaction.atomic(): gc = GiftCard.objects.select_for_update().get(pk=gcpk)
try: if gc.currency != self.event.currency: # noqa - just a safeguard
gc = GiftCard.objects.select_for_update(of=OF_SELF).get(pk=gcpk) raise PaymentException(_("This gift card does not support this currency."))
except GiftCard.DoesNotExist: if not gc.accepted_by(self.event.organizer): # noqa - just a safeguard
raise PaymentException(_("This gift card does not support this currency.")) raise PaymentException(_("This gift card is not accepted by this event organizer."))
if gc.currency != self.event.currency: # noqa - just a safeguard if payment.amount > gc.value: # noqa - just a safeguard
raise PaymentException(_("This gift card does not support this currency.")) raise PaymentException(_("This gift card was used in the meantime. Please try again."))
if not gc.accepted_by(self.event.organizer): if gc.expires and gc.expires < now(): # noqa - just a safeguard
raise PaymentException(_("This gift card is not accepted by this event organizer.")) messages.error(request, _("This gift card is no longer valid."))
if payment.amount > gc.value: return
raise PaymentException(_("This gift card was used in the meantime. Please try again.")) trans = gc.transactions.create(
if gc.testmode and not payment.order.testmode: value=-1 * payment.amount,
raise PaymentException(_("This gift card can only be used in test mode.")) order=payment.order,
if not gc.testmode and payment.order.testmode: payment=payment
raise PaymentException(_("Only test gift cards can be used in test mode.")) )
if gc.expires and gc.expires < now(): payment.info_data = {
raise PaymentException(_("This gift card is no longer valid.")) 'gift_card': gc.pk,
'transaction_id': trans.pk,
trans = gc.transactions.create( }
value=-1 * payment.amount, payment.confirm()
order=payment.order,
payment=payment
)
payment.info_data = {
'gift_card': gc.pk,
'transaction_id': trans.pk,
}
payment.confirm(send_mail=not is_early_special_case, generate_invoice=not is_early_special_case)
except PaymentException as e:
payment.fail(info={'error': str(e)})
raise e
def payment_is_valid_session(self, request: HttpRequest) -> bool: def payment_is_valid_session(self, request: HttpRequest) -> bool:
return True return True

View File

@@ -35,27 +35,22 @@
import copy import copy
import hashlib import hashlib
import itertools import itertools
import json
import logging import logging
import os import os
import re import re
import subprocess import subprocess
import tempfile import tempfile
import unicodedata
import uuid import uuid
from collections import OrderedDict from collections import OrderedDict
from functools import partial from functools import partial
from io import BytesIO from io import BytesIO
import jsonschema
from arabic_reshaper import ArabicReshaper from arabic_reshaper import ArabicReshaper
from bidi.algorithm import get_display from bidi.algorithm import get_display
from django.conf import settings from django.conf import settings
from django.contrib.staticfiles import finders from django.contrib.staticfiles import finders
from django.core.exceptions import ValidationError
from django.db.models import Max, Min from django.db.models import Max, Min
from django.dispatch import receiver from django.dispatch import receiver
from django.utils.deconstruct import deconstructible
from django.utils.formats import date_format from django.utils.formats import date_format
from django.utils.functional import SimpleLazyObject from django.utils.functional import SimpleLazyObject
from django.utils.html import conditional_escape from django.utils.html import conditional_escape
@@ -744,14 +739,12 @@ class Renderer:
if o['content'] == 'other' or o['content'] == 'other_i18n': if o['content'] == 'other' or o['content'] == 'other_i18n':
if o['content'] == 'other_i18n': if o['content'] == 'other_i18n':
text = str(LazyI18nString(o.get('text_i18n', {}))) text = str(LazyI18nString(o['text_i18n']))
else: else:
text = o.get('text', '') text = o['text']
def replace(x): def replace(x):
if x.group(1).startswith('itemmeta:'): if x.group(1).startswith('itemmeta:'):
if op.variation_id:
return op.variation.meta_data.get(x.group(1)[9:]) or ''
return op.item.meta_data.get(x.group(1)[9:]) or '' return op.item.meta_data.get(x.group(1)[9:]) or ''
elif x.group(1).startswith('meta:'): elif x.group(1).startswith('meta:'):
return ev.meta_data.get(x.group(1)[5:]) or '' return ev.meta_data.get(x.group(1)[5:]) or ''
@@ -772,8 +765,6 @@ class Renderer:
return re.sub(r'\{([a-zA-Z0-9:_]+)\}', replace, text) return re.sub(r'\{([a-zA-Z0-9:_]+)\}', replace, text)
elif o['content'].startswith('itemmeta:'): elif o['content'].startswith('itemmeta:'):
if op.variation_id:
return op.variation.meta_data.get(o['content'][9:]) or ''
return op.item.meta_data.get(o['content'][9:]) or '' return op.item.meta_data.get(o['content'][9:]) or ''
elif o['content'].startswith('meta:'): elif o['content'].startswith('meta:'):
@@ -836,13 +827,6 @@ class Renderer:
if o['italic']: if o['italic']:
font += ' I' font += ' I'
try:
ad = getAscentDescent(font, float(o['fontsize']))
except KeyError: # font not known, fall back
logger.warning(f'Use of unknown font "{font}"')
font = 'Open Sans'
ad = getAscentDescent(font, float(o['fontsize']))
align_map = { align_map = {
'left': TA_LEFT, 'left': TA_LEFT,
'center': TA_CENTER, 'center': TA_CENTER,
@@ -869,12 +853,10 @@ class Renderer:
except: except:
logger.exception('Reshaping/Bidi fixes failed on string {}'.format(repr(text))) logger.exception('Reshaping/Bidi fixes failed on string {}'.format(repr(text)))
# reportlab does not support unicode combination characters
text = unicodedata.normalize("NFKC", text)
p = Paragraph(text, style=style) p = Paragraph(text, style=style)
w, h = p.wrapOn(canvas, float(o['width']) * mm, 1000 * mm) w, h = p.wrapOn(canvas, float(o['width']) * mm, 1000 * mm)
# p_size = p.wrap(float(o['width']) * mm, 1000 * mm) # p_size = p.wrap(float(o['width']) * mm, 1000 * mm)
ad = getAscentDescent(font, float(o['fontsize']))
canvas.saveState() canvas.saveState()
# The ascent/descent offsets here are not really proven to be correct, they're just empirical values to get # The ascent/descent offsets here are not really proven to be correct, they're just empirical values to get
# reportlab render similarly to browser canvas. # reportlab render similarly to browser canvas.
@@ -979,22 +961,3 @@ class Renderer:
output.write(outbuffer) output.write(outbuffer)
outbuffer.seek(0) outbuffer.seek(0)
return outbuffer return outbuffer
@deconstructible
class PdfLayoutValidator:
def __call__(self, value):
if not isinstance(value, dict):
try:
val = json.loads(value)
except ValueError:
raise ValidationError(_('Your layout file is not a valid JSON file.'))
else:
val = value
with open(finders.find('schema/pdf-layout.schema.json'), 'r') as f:
schema = json.loads(f.read())
try:
jsonschema.validate(val, schema)
except jsonschema.ValidationError as e:
e = str(e).replace('%', '%%')
raise ValidationError(_('Your layout file is not a valid layout. Error message: {}').format(e))

View File

@@ -65,14 +65,7 @@ def get_all_plugins(event=None) -> List[type]:
) )
class PluginConfigMeta(type): class PluginConfig(AppConfig):
def __getattribute__(cls, item):
if item == "default" and cls is PluginConfig:
return False
return super().__getattribute__(item)
class PluginConfig(AppConfig, metaclass=PluginConfigMeta):
IGNORE = False IGNORE = False
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):

Some files were not shown because too many files have changed in this diff Show More