mirror of
https://github.com/TandoorRecipes/recipes.git
synced 2026-01-07 23:28:16 -05:00
Merge branch 'develop' into feature/2402-make-now-count
This commit is contained in:
@@ -3,7 +3,6 @@ npm-debug.log
|
|||||||
Dockerfile*
|
Dockerfile*
|
||||||
docker-compose*
|
docker-compose*
|
||||||
.dockerignore
|
.dockerignore
|
||||||
.git
|
|
||||||
.gitignore
|
.gitignore
|
||||||
README.md
|
README.md
|
||||||
LICENSE
|
LICENSE
|
||||||
|
|||||||
@@ -2,6 +2,10 @@
|
|||||||
# when unset: 1 (true) - dont unset this, just for development
|
# when unset: 1 (true) - dont unset this, just for development
|
||||||
DEBUG=0
|
DEBUG=0
|
||||||
SQL_DEBUG=0
|
SQL_DEBUG=0
|
||||||
|
DEBUG_TOOLBAR=0
|
||||||
|
# Gunicorn log level for debugging (default value is "info" when unset)
|
||||||
|
# (see https://docs.gunicorn.org/en/stable/settings.html#loglevel for available settings)
|
||||||
|
# GUNICORN_LOG_LEVEL="debug"
|
||||||
|
|
||||||
# HTTP port to bind to
|
# HTTP port to bind to
|
||||||
# TANDOOR_PORT=8080
|
# TANDOOR_PORT=8080
|
||||||
@@ -9,9 +13,18 @@ SQL_DEBUG=0
|
|||||||
# hosts the application can run under e.g. recipes.mydomain.com,cooking.mydomain.com,...
|
# hosts the application can run under e.g. recipes.mydomain.com,cooking.mydomain.com,...
|
||||||
ALLOWED_HOSTS=*
|
ALLOWED_HOSTS=*
|
||||||
|
|
||||||
|
# Cross Site Request Forgery protection
|
||||||
|
# (https://docs.djangoproject.com/en/4.2/ref/settings/#std-setting-CSRF_TRUSTED_ORIGINS)
|
||||||
|
# CSRF_TRUSTED_ORIGINS = []
|
||||||
|
|
||||||
|
# Cross Origin Resource Sharing
|
||||||
|
# (https://github.com/adamchainz/django-cors-header)
|
||||||
|
# CORS_ALLOW_ALL_ORIGINS = True
|
||||||
|
|
||||||
# random secret key, use for example `base64 /dev/urandom | head -c50` to generate one
|
# random secret key, use for example `base64 /dev/urandom | head -c50` to generate one
|
||||||
# ---------------------------- REQUIRED -------------------------
|
# ---------------------------- AT LEAST ONE REQUIRED -------------------------
|
||||||
SECRET_KEY=
|
SECRET_KEY=
|
||||||
|
SECRET_KEY_FILE=
|
||||||
# ---------------------------------------------------------------
|
# ---------------------------------------------------------------
|
||||||
|
|
||||||
# your default timezone See https://timezonedb.com/time-zones for a list of timezones
|
# your default timezone See https://timezonedb.com/time-zones for a list of timezones
|
||||||
@@ -23,8 +36,9 @@ DB_ENGINE=django.db.backends.postgresql
|
|||||||
POSTGRES_HOST=db_recipes
|
POSTGRES_HOST=db_recipes
|
||||||
POSTGRES_PORT=5432
|
POSTGRES_PORT=5432
|
||||||
POSTGRES_USER=djangouser
|
POSTGRES_USER=djangouser
|
||||||
# ---------------------------- REQUIRED -------------------------
|
# ---------------------------- AT LEAST ONE REQUIRED -------------------------
|
||||||
POSTGRES_PASSWORD=
|
POSTGRES_PASSWORD=
|
||||||
|
POSTGRES_PASSWORD_FILE=
|
||||||
# ---------------------------------------------------------------
|
# ---------------------------------------------------------------
|
||||||
POSTGRES_DB=djangodb
|
POSTGRES_DB=djangodb
|
||||||
|
|
||||||
@@ -96,10 +110,12 @@ GUNICORN_MEDIA=0
|
|||||||
# prefix used for account related emails (default "[Tandoor Recipes] ")
|
# prefix used for account related emails (default "[Tandoor Recipes] ")
|
||||||
# ACCOUNT_EMAIL_SUBJECT_PREFIX=
|
# ACCOUNT_EMAIL_SUBJECT_PREFIX=
|
||||||
|
|
||||||
# allow authentication via reverse proxy (e.g. authelia), leave off if you dont know what you are doing
|
# allow authentication via the REMOTE-USER header (can be used for e.g. authelia).
|
||||||
# see docs for more information https://docs.tandoor.dev/features/authentication/
|
# ATTENTION: Leave off if you don't know what you are doing! Enabling this without proper configuration will enable anybody
|
||||||
|
# to login with any username!
|
||||||
|
# See docs for additional information: https://docs.tandoor.dev/features/authentication/#reverse-proxy-authentication
|
||||||
# when unset: 0 (false)
|
# when unset: 0 (false)
|
||||||
REVERSE_PROXY_AUTH=0
|
REMOTE_USER_AUTH=0
|
||||||
|
|
||||||
# Default settings for spaces, apply per space and can be changed in the admin view
|
# Default settings for spaces, apply per space and can be changed in the admin view
|
||||||
# SPACE_DEFAULT_MAX_RECIPES=0 # 0=unlimited recipes
|
# SPACE_DEFAULT_MAX_RECIPES=0 # 0=unlimited recipes
|
||||||
@@ -107,7 +123,8 @@ REVERSE_PROXY_AUTH=0
|
|||||||
# SPACE_DEFAULT_MAX_FILES=0 # Maximum file storage for space in MB. 0 for unlimited, -1 to disable file upload.
|
# SPACE_DEFAULT_MAX_FILES=0 # Maximum file storage for space in MB. 0 for unlimited, -1 to disable file upload.
|
||||||
# SPACE_DEFAULT_ALLOW_SHARING=1 # Allow users to share recipes with public links
|
# SPACE_DEFAULT_ALLOW_SHARING=1 # Allow users to share recipes with public links
|
||||||
|
|
||||||
# allow people to create accounts on your application instance (without an invite link)
|
# allow people to create local accounts on your application instance (without an invite link)
|
||||||
|
# social accounts will always be able to sign up
|
||||||
# when unset: 0 (false)
|
# when unset: 0 (false)
|
||||||
# ENABLE_SIGNUP=0
|
# ENABLE_SIGNUP=0
|
||||||
|
|
||||||
@@ -157,6 +174,7 @@ REVERSE_PROXY_AUTH=0
|
|||||||
#AUTH_LDAP_BIND_PASSWORD=
|
#AUTH_LDAP_BIND_PASSWORD=
|
||||||
#AUTH_LDAP_USER_SEARCH_BASE_DN=
|
#AUTH_LDAP_USER_SEARCH_BASE_DN=
|
||||||
#AUTH_LDAP_TLS_CACERTFILE=
|
#AUTH_LDAP_TLS_CACERTFILE=
|
||||||
|
#AUTH_LDAP_START_TLS=
|
||||||
|
|
||||||
# Enables exporting PDF (see export docs)
|
# Enables exporting PDF (see export docs)
|
||||||
# Disabled by default, uncomment to enable
|
# Disabled by default, uncomment to enable
|
||||||
|
|||||||
110
.github/workflows/build-docker-open-data.yml
vendored
Normal file
110
.github/workflows/build-docker-open-data.yml
vendored
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
name: Build Docker Container with open data plugin installed
|
||||||
|
|
||||||
|
on: push
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-container:
|
||||||
|
name: Build ${{ matrix.name }} Container
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.repository_owner == 'TandoorRecipes'
|
||||||
|
continue-on-error: ${{ matrix.continue-on-error }}
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
packages: write
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
# Standard build config
|
||||||
|
- name: Standard
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
suffix: ""
|
||||||
|
continue-on-error: false
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Get version number
|
||||||
|
id: get_version
|
||||||
|
run: |
|
||||||
|
if [[ "$GITHUB_REF" = refs/tags/* ]]; then
|
||||||
|
echo "VERSION=${GITHUB_REF/refs\/tags\//}" >> $GITHUB_OUTPUT
|
||||||
|
elif [[ "$GITHUB_REF" = refs/heads/beta ]]; then
|
||||||
|
echo VERSION=beta >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo VERSION=develop >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
|
||||||
|
# clone open data plugin
|
||||||
|
- name: clone open data plugin repo
|
||||||
|
uses: actions/checkout@master
|
||||||
|
with:
|
||||||
|
repository: TandoorRecipes/open_data_plugin
|
||||||
|
ref: master
|
||||||
|
path: ./recipes/plugins/open_data_plugin
|
||||||
|
|
||||||
|
# Build Vue frontend
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: '18'
|
||||||
|
cache: yarn
|
||||||
|
cache-dependency-path: vue/yarn.lock
|
||||||
|
- name: Install dependencies
|
||||||
|
working-directory: ./vue
|
||||||
|
run: yarn install --frozen-lockfile
|
||||||
|
- name: Build dependencies
|
||||||
|
working-directory: ./vue
|
||||||
|
run: yarn build
|
||||||
|
|
||||||
|
- name: Setup Open Data Plugin Links
|
||||||
|
working-directory: ./recipes/plugins/open_data_plugin
|
||||||
|
run: python setup_repo.py
|
||||||
|
|
||||||
|
- name: Build Open Data Frontend
|
||||||
|
working-directory: ./recipes/plugins/open_data_plugin/vue
|
||||||
|
run: yarn build
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v2
|
||||||
|
- name: Set up Buildx
|
||||||
|
uses: docker/setup-buildx-action@v2
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v2
|
||||||
|
if: github.secret_source == 'Actions'
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKER_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
- name: Login to GitHub Container Registry
|
||||||
|
uses: docker/login-action@v2
|
||||||
|
if: github.secret_source == 'Actions'
|
||||||
|
with:
|
||||||
|
registry: ghcr.io
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ github.token }}
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v4
|
||||||
|
with:
|
||||||
|
images: |
|
||||||
|
vabene1111/recipes
|
||||||
|
ghcr.io/TandoorRecipes/recipes
|
||||||
|
flavor: |
|
||||||
|
latest=false
|
||||||
|
suffix=${{ matrix.suffix }}
|
||||||
|
tags: |
|
||||||
|
type=raw,value=latest,suffix=-open-data-plugin,enable=${{ startsWith(github.ref, 'refs/tags/') }}
|
||||||
|
type=semver,suffix=-open-data-plugin,pattern={{version}}
|
||||||
|
type=semver,suffix=-open-data-plugin,pattern={{major}}.{{minor}}
|
||||||
|
type=semver,suffix=-open-data-plugin,pattern={{major}}
|
||||||
|
type=ref,suffix=-open-data-plugin,event=branch
|
||||||
|
- name: Build and Push
|
||||||
|
uses: docker/build-push-action@v4
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
file: ${{ matrix.dockerfile }}
|
||||||
|
pull: true
|
||||||
|
push: ${{ github.secret_source == 'Actions' }}
|
||||||
|
platforms: ${{ matrix.platforms }}
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
26
.github/workflows/build-docker.yml
vendored
26
.github/workflows/build-docker.yml
vendored
@@ -17,15 +17,9 @@ jobs:
|
|||||||
# Standard build config
|
# Standard build config
|
||||||
- name: Standard
|
- name: Standard
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
platforms: linux/amd64,linux/arm64
|
platforms: linux/amd64,linux/arm64,linux/arm/v7
|
||||||
suffix: ""
|
suffix: ""
|
||||||
continue-on-error: false
|
continue-on-error: false
|
||||||
# Raspi build config
|
|
||||||
- name: Raspi
|
|
||||||
dockerfile: Dockerfile-raspi
|
|
||||||
platforms: linux/arm/v7
|
|
||||||
suffix: "-raspi"
|
|
||||||
continue-on-error: true
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
@@ -40,20 +34,10 @@ jobs:
|
|||||||
echo VERSION=develop >> $GITHUB_OUTPUT
|
echo VERSION=develop >> $GITHUB_OUTPUT
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Update Version number
|
|
||||||
- name: Update version file
|
|
||||||
uses: DamianReeves/write-file-action@v1.2
|
|
||||||
with:
|
|
||||||
path: recipes/version.py
|
|
||||||
contents: |
|
|
||||||
VERSION_NUMBER = '${{ steps.get_version.outputs.VERSION }}'
|
|
||||||
BUILD_REF = '${{ github.sha }}'
|
|
||||||
write-mode: overwrite
|
|
||||||
|
|
||||||
# Build Vue frontend
|
# Build Vue frontend
|
||||||
- uses: actions/setup-node@v3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: '14'
|
node-version: '18'
|
||||||
cache: yarn
|
cache: yarn
|
||||||
cache-dependency-path: vue/yarn.lock
|
cache-dependency-path: vue/yarn.lock
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
@@ -115,13 +99,17 @@ jobs:
|
|||||||
needs: build-container
|
needs: build-container
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
steps:
|
steps:
|
||||||
|
- name: Set tag name
|
||||||
|
run: |
|
||||||
|
# Strip "refs/tags/" prefix
|
||||||
|
echo "VERSION=${GITHUB_REF#refs/tags/}" >> $GITHUB_ENV
|
||||||
# Send stable discord notification
|
# Send stable discord notification
|
||||||
- name: Discord notification
|
- name: Discord notification
|
||||||
env:
|
env:
|
||||||
DISCORD_WEBHOOK: ${{ secrets.DISCORD_RELEASE_WEBHOOK }}
|
DISCORD_WEBHOOK: ${{ secrets.DISCORD_RELEASE_WEBHOOK }}
|
||||||
uses: Ilshidur/action-discord@0.3.2
|
uses: Ilshidur/action-discord@0.3.2
|
||||||
with:
|
with:
|
||||||
args: '🚀 Version {{ EVENT_PAYLOAD.release.tag_name }} of tandoor has been released 🥳 Check it out https://github.com/vabene1111/recipes/releases/tag/{{ EVENT_PAYLOAD.release.tag_name }}'
|
args: '🚀 Version {{ VERSION }} of tandoor has been released 🥳 Check it out https://github.com/vabene1111/recipes/releases/tag/{{ VERSION }}'
|
||||||
|
|
||||||
notify-beta:
|
notify-beta:
|
||||||
name: Notify Beta
|
name: Notify Beta
|
||||||
|
|||||||
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
@@ -20,7 +20,7 @@ jobs:
|
|||||||
# Build Vue frontend
|
# Build Vue frontend
|
||||||
- uses: actions/setup-node@v3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: '16'
|
node-version: '18'
|
||||||
- name: Install Vue dependencies
|
- name: Install Vue dependencies
|
||||||
working-directory: ./vue
|
working-directory: ./vue
|
||||||
run: yarn install
|
run: yarn install
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -78,6 +78,7 @@ postgresql/
|
|||||||
|
|
||||||
/docker-compose.override.yml
|
/docker-compose.override.yml
|
||||||
vue/node_modules
|
vue/node_modules
|
||||||
|
plugins
|
||||||
.vscode/
|
.vscode/
|
||||||
vetur.config.js
|
vetur.config.js
|
||||||
cookbook/static/vue
|
cookbook/static/vue
|
||||||
|
|||||||
2
.idea/recipes.iml
generated
2
.idea/recipes.iml
generated
@@ -18,7 +18,7 @@
|
|||||||
<excludeFolder url="file://$MODULE_DIR$/staticfiles" />
|
<excludeFolder url="file://$MODULE_DIR$/staticfiles" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/venv" />
|
<excludeFolder url="file://$MODULE_DIR$/venv" />
|
||||||
</content>
|
</content>
|
||||||
<orderEntry type="jdk" jdkName="Python 3.11 (recipes)" jdkType="Python SDK" />
|
<orderEntry type="inheritedJdk" />
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
</component>
|
</component>
|
||||||
<component name="TemplatesService">
|
<component name="TemplatesService">
|
||||||
|
|||||||
2
.idea/vcs.xml
generated
2
.idea/vcs.xml
generated
@@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="VcsDirectoryMappings">
|
<component name="VcsDirectoryMappings">
|
||||||
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
<mapping directory="" vcs="Git" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
||||||
16
Dockerfile
16
Dockerfile
@@ -1,7 +1,7 @@
|
|||||||
FROM python:3.10-alpine3.15
|
FROM python:3.10-alpine3.18
|
||||||
|
|
||||||
#Install all dependencies.
|
#Install all dependencies.
|
||||||
RUN apk add --no-cache postgresql-libs postgresql-client gettext zlib libjpeg libwebp libxml2-dev libxslt-dev py-cryptography openldap
|
RUN apk add --no-cache postgresql-libs postgresql-client gettext zlib libjpeg libwebp libxml2-dev libxslt-dev openldap git
|
||||||
|
|
||||||
#Print all logs without buffering it.
|
#Print all logs without buffering it.
|
||||||
ENV PYTHONUNBUFFERED 1
|
ENV PYTHONUNBUFFERED 1
|
||||||
@@ -15,7 +15,11 @@ WORKDIR /opt/recipes
|
|||||||
|
|
||||||
COPY requirements.txt ./
|
COPY requirements.txt ./
|
||||||
|
|
||||||
RUN apk add --no-cache --virtual .build-deps gcc musl-dev postgresql-dev zlib-dev jpeg-dev libwebp-dev openssl-dev libffi-dev cargo openldap-dev python3-dev git && \
|
RUN \
|
||||||
|
if [ `apk --print-arch` = "armv7" ]; then \
|
||||||
|
printf "[global]\nextra-index-url=https://www.piwheels.org/simple\n" > /etc/pip.conf ; \
|
||||||
|
fi
|
||||||
|
RUN apk add --no-cache --virtual .build-deps gcc musl-dev postgresql-dev zlib-dev jpeg-dev libwebp-dev openssl-dev libffi-dev cargo openldap-dev python3-dev && \
|
||||||
echo -n "INPUT ( libldap.so )" > /usr/lib/libldap_r.so && \
|
echo -n "INPUT ( libldap.so )" > /usr/lib/libldap_r.so && \
|
||||||
python -m venv venv && \
|
python -m venv venv && \
|
||||||
/opt/recipes/venv/bin/python -m pip install --upgrade pip && \
|
/opt/recipes/venv/bin/python -m pip install --upgrade pip && \
|
||||||
@@ -26,5 +30,11 @@ RUN apk add --no-cache --virtual .build-deps gcc musl-dev postgresql-dev zlib-de
|
|||||||
|
|
||||||
#Copy project and execute it.
|
#Copy project and execute it.
|
||||||
COPY . ./
|
COPY . ./
|
||||||
|
|
||||||
|
# collect information from git repositories
|
||||||
|
RUN /opt/recipes/venv/bin/python version.py
|
||||||
|
# delete git repositories to reduce image size
|
||||||
|
RUN find . -type d -name ".git" | xargs rm -rf
|
||||||
|
|
||||||
RUN chmod +x boot.sh
|
RUN chmod +x boot.sh
|
||||||
ENTRYPOINT ["/opt/recipes/boot.sh"]
|
ENTRYPOINT ["/opt/recipes/boot.sh"]
|
||||||
|
|||||||
@@ -1,33 +0,0 @@
|
|||||||
# builds of cryptography for raspberry pi (or better arm v7) fail for some
|
|
||||||
FROM python:3.9-alpine3.15
|
|
||||||
|
|
||||||
#Install all dependencies.
|
|
||||||
RUN apk add --no-cache postgresql-libs postgresql-client gettext zlib libjpeg libwebp libxml2-dev libxslt-dev py-cryptography openldap gcompat
|
|
||||||
|
|
||||||
#Print all logs without buffering it.
|
|
||||||
ENV PYTHONUNBUFFERED 1
|
|
||||||
|
|
||||||
#This port will be used by gunicorn.
|
|
||||||
EXPOSE 8080
|
|
||||||
|
|
||||||
#Create app dir and install requirements.
|
|
||||||
RUN mkdir /opt/recipes
|
|
||||||
WORKDIR /opt/recipes
|
|
||||||
|
|
||||||
COPY requirements.txt ./
|
|
||||||
RUN \
|
|
||||||
if [ `apk --print-arch` = "armv7" ]; then \
|
|
||||||
printf "[global]\nextra-index-url=https://www.piwheels.org/simple\n" > /etc/pip.conf ; \
|
|
||||||
fi
|
|
||||||
RUN apk add --no-cache --virtual .build-deps gcc musl-dev zlib-dev jpeg-dev libwebp-dev python3-dev git && \
|
|
||||||
echo -n "INPUT ( libldap.so )" > /usr/lib/libldap_r.so && \
|
|
||||||
python -m venv venv && \
|
|
||||||
/opt/recipes/venv/bin/python -m pip install --upgrade pip && \
|
|
||||||
venv/bin/pip install wheel==0.37.1 && \
|
|
||||||
venv/bin/pip install -r requirements.txt --no-cache-dir --no-binary=Pillow && \
|
|
||||||
apk --purge del .build-deps
|
|
||||||
|
|
||||||
#Copy project and execute it.
|
|
||||||
COPY . ./
|
|
||||||
RUN chmod +x boot.sh
|
|
||||||
ENTRYPOINT ["/opt/recipes/boot.sh"]
|
|
||||||
21
boot.sh
21
boot.sh
@@ -4,6 +4,7 @@ source venv/bin/activate
|
|||||||
TANDOOR_PORT="${TANDOOR_PORT:-8080}"
|
TANDOOR_PORT="${TANDOOR_PORT:-8080}"
|
||||||
GUNICORN_WORKERS="${GUNICORN_WORKERS:-3}"
|
GUNICORN_WORKERS="${GUNICORN_WORKERS:-3}"
|
||||||
GUNICORN_THREADS="${GUNICORN_THREADS:-2}"
|
GUNICORN_THREADS="${GUNICORN_THREADS:-2}"
|
||||||
|
GUNICORN_LOG_LEVEL="${GUNICORN_LOG_LEVEL:-'info'}"
|
||||||
NGINX_CONF_FILE=/opt/recipes/nginx/conf.d/Recipes.conf
|
NGINX_CONF_FILE=/opt/recipes/nginx/conf.d/Recipes.conf
|
||||||
|
|
||||||
display_warning() {
|
display_warning() {
|
||||||
@@ -18,9 +19,14 @@ if [ ! -f "$NGINX_CONF_FILE" ] && [ $GUNICORN_MEDIA -eq 0 ]; then
|
|||||||
display_warning "Nginx configuration file could not be found at the default location!\nPath: ${NGINX_CONF_FILE}"
|
display_warning "Nginx configuration file could not be found at the default location!\nPath: ${NGINX_CONF_FILE}"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# SECRET_KEY must be set in .env file
|
# SECRET_KEY (or a valid file at SECRET_KEY_FILE) must be set in .env file
|
||||||
|
|
||||||
|
if [ -f "${SECRET_KEY_FILE}" ]; then
|
||||||
|
export SECRET_KEY=$(cat "$SECRET_KEY_FILE")
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -z "${SECRET_KEY}" ]; then
|
if [ -z "${SECRET_KEY}" ]; then
|
||||||
display_warning "The environment variable 'SECRET_KEY' is not set but REQUIRED for running Tandoor!"
|
display_warning "The environment variable 'SECRET_KEY' (or 'SECRET_KEY_FILE' that points to an existing file) is not set but REQUIRED for running Tandoor!"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|
||||||
@@ -31,9 +37,14 @@ max_attempts=20
|
|||||||
|
|
||||||
if [ "${DB_ENGINE}" != 'django.db.backends.sqlite3' ]; then
|
if [ "${DB_ENGINE}" != 'django.db.backends.sqlite3' ]; then
|
||||||
|
|
||||||
# POSTGRES_PASSWORD must be set in .env file
|
# POSTGRES_PASSWORD (or a valid file at POSTGRES_PASSWORD_FILE) must be set in .env file
|
||||||
|
|
||||||
|
if [ -f "${POSTGRES_PASSWORD_FILE}" ]; then
|
||||||
|
export POSTGRES_PASSWORD=$(cat "$POSTGRES_PASSWORD_FILE")
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -z "${POSTGRES_PASSWORD}" ]; then
|
if [ -z "${POSTGRES_PASSWORD}" ]; then
|
||||||
display_warning "The environment variable 'POSTGRES_PASSWORD' is not set but REQUIRED for running Tandoor!"
|
display_warning "The environment variable 'POSTGRES_PASSWORD' (or 'POSTGRES_PASSWORD_FILE' that points to an existing file) is not set but REQUIRED for running Tandoor!"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
while pg_isready --host=${POSTGRES_HOST} --port=${POSTGRES_PORT} --user=${POSTGRES_USER} -q; status=$?; attempt=$((attempt+1)); [ $status -ne 0 ] && [ $attempt -le $max_attempts ]; do
|
while pg_isready --host=${POSTGRES_HOST} --port=${POSTGRES_PORT} --user=${POSTGRES_USER} -q; status=$?; attempt=$((attempt+1)); [ $status -ne 0 ] && [ $attempt -le $max_attempts ]; do
|
||||||
@@ -65,4 +76,4 @@ echo "Done"
|
|||||||
|
|
||||||
chmod -R 755 /opt/recipes/mediafiles
|
chmod -R 755 /opt/recipes/mediafiles
|
||||||
|
|
||||||
exec gunicorn -b :$TANDOOR_PORT --workers $GUNICORN_WORKERS --threads $GUNICORN_THREADS --access-logfile - --error-logfile - --log-level INFO recipes.wsgi
|
exec gunicorn -b :$TANDOOR_PORT --workers $GUNICORN_WORKERS --threads $GUNICORN_THREADS --access-logfile - --error-logfile - --log-level $GUNICORN_LOG_LEVEL recipes.wsgi
|
||||||
|
|||||||
@@ -10,12 +10,13 @@ from treebeard.forms import movenodeform_factory
|
|||||||
|
|
||||||
from cookbook.managers import DICTIONARY
|
from cookbook.managers import DICTIONARY
|
||||||
|
|
||||||
from .models import (BookmarkletImport, Comment, CookLog, Food, FoodInheritField, ImportLog,
|
from .models import (BookmarkletImport, Comment, CookLog, Food, ImportLog, Ingredient, InviteLink,
|
||||||
Ingredient, InviteLink, Keyword, MealPlan, MealType, NutritionInformation,
|
Keyword, MealPlan, MealType, NutritionInformation, Property, PropertyType,
|
||||||
Recipe, RecipeBook, RecipeBookEntry, RecipeImport, SearchPreference, ShareLink,
|
Recipe, RecipeBook, RecipeBookEntry, RecipeImport, SearchPreference, ShareLink,
|
||||||
ShoppingList, ShoppingListEntry, ShoppingListRecipe, Space, Step, Storage,
|
ShoppingList, ShoppingListEntry, ShoppingListRecipe, Space, Step, Storage,
|
||||||
Supermarket, SupermarketCategory, SupermarketCategoryRelation, Sync, SyncLog,
|
Supermarket, SupermarketCategory, SupermarketCategoryRelation, Sync, SyncLog,
|
||||||
TelegramBot, Unit, UserFile, UserPreference, ViewLog, Automation, UserSpace)
|
TelegramBot, Unit, UnitConversion, UserFile, UserPreference, UserSpace,
|
||||||
|
ViewLog)
|
||||||
|
|
||||||
|
|
||||||
class CustomUserAdmin(UserAdmin):
|
class CustomUserAdmin(UserAdmin):
|
||||||
@@ -38,6 +39,8 @@ def delete_space_action(modeladmin, request, queryset):
|
|||||||
class SpaceAdmin(admin.ModelAdmin):
|
class SpaceAdmin(admin.ModelAdmin):
|
||||||
list_display = ('name', 'created_by', 'max_recipes', 'max_users', 'max_file_storage_mb', 'allow_sharing')
|
list_display = ('name', 'created_by', 'max_recipes', 'max_users', 'max_file_storage_mb', 'allow_sharing')
|
||||||
search_fields = ('name', 'created_by__username')
|
search_fields = ('name', 'created_by__username')
|
||||||
|
autocomplete_fields = ('created_by',)
|
||||||
|
filter_horizontal = ('food_inherit',)
|
||||||
list_filter = ('max_recipes', 'max_users', 'max_file_storage_mb', 'allow_sharing')
|
list_filter = ('max_recipes', 'max_users', 'max_file_storage_mb', 'allow_sharing')
|
||||||
date_hierarchy = 'created_at'
|
date_hierarchy = 'created_at'
|
||||||
actions = [delete_space_action]
|
actions = [delete_space_action]
|
||||||
@@ -49,6 +52,8 @@ admin.site.register(Space, SpaceAdmin)
|
|||||||
class UserSpaceAdmin(admin.ModelAdmin):
|
class UserSpaceAdmin(admin.ModelAdmin):
|
||||||
list_display = ('user', 'space',)
|
list_display = ('user', 'space',)
|
||||||
search_fields = ('user__username', 'space__name',)
|
search_fields = ('user__username', 'space__name',)
|
||||||
|
filter_horizontal = ('groups',)
|
||||||
|
autocomplete_fields = ('user', 'space',)
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(UserSpace, UserSpaceAdmin)
|
admin.site.register(UserSpace, UserSpaceAdmin)
|
||||||
@@ -59,6 +64,7 @@ class UserPreferenceAdmin(admin.ModelAdmin):
|
|||||||
search_fields = ('user__username',)
|
search_fields = ('user__username',)
|
||||||
list_filter = ('theme', 'nav_color', 'default_page',)
|
list_filter = ('theme', 'nav_color', 'default_page',)
|
||||||
date_hierarchy = 'created_at'
|
date_hierarchy = 'created_at'
|
||||||
|
filter_horizontal = ('plan_share', 'shopping_share',)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def name(obj):
|
def name(obj):
|
||||||
@@ -150,9 +156,16 @@ class KeywordAdmin(TreeAdmin):
|
|||||||
admin.site.register(Keyword, KeywordAdmin)
|
admin.site.register(Keyword, KeywordAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.action(description='Delete Steps not part of a Recipe.')
|
||||||
|
def delete_unattached_steps(modeladmin, request, queryset):
|
||||||
|
with scopes_disabled():
|
||||||
|
Step.objects.filter(recipe=None).delete()
|
||||||
|
|
||||||
|
|
||||||
class StepAdmin(admin.ModelAdmin):
|
class StepAdmin(admin.ModelAdmin):
|
||||||
list_display = ('name', 'order',)
|
list_display = ('name', 'order',)
|
||||||
search_fields = ('name',)
|
search_fields = ('name',)
|
||||||
|
actions = [delete_unattached_steps]
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(Step, StepAdmin)
|
admin.site.register(Step, StepAdmin)
|
||||||
@@ -201,9 +214,24 @@ class FoodAdmin(TreeAdmin):
|
|||||||
admin.site.register(Food, FoodAdmin)
|
admin.site.register(Food, FoodAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
class UnitConversionAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('base_amount', 'base_unit', 'food', 'converted_amount', 'converted_unit')
|
||||||
|
search_fields = ('food__name', 'unit__name')
|
||||||
|
|
||||||
|
|
||||||
|
admin.site.register(UnitConversion, UnitConversionAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.action(description='Delete Ingredients not part of a Recipe.')
|
||||||
|
def delete_unattached_ingredients(modeladmin, request, queryset):
|
||||||
|
with scopes_disabled():
|
||||||
|
Ingredient.objects.filter(step__recipe=None).delete()
|
||||||
|
|
||||||
|
|
||||||
class IngredientAdmin(admin.ModelAdmin):
|
class IngredientAdmin(admin.ModelAdmin):
|
||||||
list_display = ('food', 'amount', 'unit')
|
list_display = ('food', 'amount', 'unit')
|
||||||
search_fields = ('food__name', 'unit__name')
|
search_fields = ('food__name', 'unit__name')
|
||||||
|
actions = [delete_unattached_ingredients]
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(Ingredient, IngredientAdmin)
|
admin.site.register(Ingredient, IngredientAdmin)
|
||||||
@@ -249,7 +277,7 @@ admin.site.register(RecipeBookEntry, RecipeBookEntryAdmin)
|
|||||||
|
|
||||||
|
|
||||||
class MealPlanAdmin(admin.ModelAdmin):
|
class MealPlanAdmin(admin.ModelAdmin):
|
||||||
list_display = ('user', 'recipe', 'meal_type', 'date')
|
list_display = ('user', 'recipe', 'meal_type', 'from_date', 'to_date')
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def user(obj):
|
def user(obj):
|
||||||
@@ -286,6 +314,7 @@ admin.site.register(InviteLink, InviteLinkAdmin)
|
|||||||
|
|
||||||
class CookLogAdmin(admin.ModelAdmin):
|
class CookLogAdmin(admin.ModelAdmin):
|
||||||
list_display = ('recipe', 'created_by', 'created_at', 'rating', 'servings')
|
list_display = ('recipe', 'created_by', 'created_at', 'rating', 'servings')
|
||||||
|
search_fields = ('recipe__name', 'space__name',)
|
||||||
|
|
||||||
|
|
||||||
admin.site.register(CookLog, CookLogAdmin)
|
admin.site.register(CookLog, CookLogAdmin)
|
||||||
@@ -319,6 +348,20 @@ class ShareLinkAdmin(admin.ModelAdmin):
|
|||||||
admin.site.register(ShareLink, ShareLinkAdmin)
|
admin.site.register(ShareLink, ShareLinkAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyTypeAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('id', 'name')
|
||||||
|
|
||||||
|
|
||||||
|
admin.site.register(PropertyType, PropertyTypeAdmin)
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ('property_amount', 'property_type')
|
||||||
|
|
||||||
|
|
||||||
|
admin.site.register(Property, PropertyAdmin)
|
||||||
|
|
||||||
|
|
||||||
class NutritionInformationAdmin(admin.ModelAdmin):
|
class NutritionInformationAdmin(admin.ModelAdmin):
|
||||||
list_display = ('id',)
|
list_display = ('id',)
|
||||||
|
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ class UserPreferenceForm(forms.ModelForm):
|
|||||||
model = UserPreference
|
model = UserPreference
|
||||||
fields = (
|
fields = (
|
||||||
'default_unit', 'use_fractions', 'use_kj', 'theme', 'nav_color',
|
'default_unit', 'use_fractions', 'use_kj', 'theme', 'nav_color',
|
||||||
'sticky_navbar', 'default_page', 'plan_share', 'ingredient_decimals', 'comments', 'left_handed',
|
'sticky_navbar', 'default_page', 'plan_share', 'ingredient_decimals', 'comments', 'left_handed', 'show_step_ingredients',
|
||||||
)
|
)
|
||||||
|
|
||||||
labels = {
|
labels = {
|
||||||
@@ -60,29 +60,29 @@ class UserPreferenceForm(forms.ModelForm):
|
|||||||
'ingredient_decimals': _('Ingredient decimal places'),
|
'ingredient_decimals': _('Ingredient decimal places'),
|
||||||
'shopping_auto_sync': _('Shopping list auto sync period'),
|
'shopping_auto_sync': _('Shopping list auto sync period'),
|
||||||
'comments': _('Comments'),
|
'comments': _('Comments'),
|
||||||
'left_handed': _('Left-handed mode')
|
'left_handed': _('Left-handed mode'),
|
||||||
|
'show_step_ingredients': _('Show step ingredients table')
|
||||||
}
|
}
|
||||||
|
|
||||||
help_texts = {
|
help_texts = {
|
||||||
'nav_color': _('Color of the top navigation bar. Not all colors work with all themes, just try them out!'),
|
'nav_color': _('Color of the top navigation bar. Not all colors work with all themes, just try them out!'),
|
||||||
|
'default_unit': _('Default Unit to be used when inserting a new ingredient into a recipe.'),
|
||||||
'default_unit': _('Default Unit to be used when inserting a new ingredient into a recipe.'),
|
|
||||||
'use_fractions': _(
|
'use_fractions': _(
|
||||||
'Enables support for fractions in ingredient amounts (e.g. convert decimals to fractions automatically)'),
|
'Enables support for fractions in ingredient amounts (e.g. convert decimals to fractions automatically)'),
|
||||||
|
'use_kj': _('Display nutritional energy amounts in joules instead of calories'),
|
||||||
'use_kj': _('Display nutritional energy amounts in joules instead of calories'),
|
|
||||||
'plan_share': _('Users with whom newly created meal plans should be shared by default.'),
|
'plan_share': _('Users with whom newly created meal plans should be shared by default.'),
|
||||||
'shopping_share': _('Users with whom to share shopping lists.'),
|
'shopping_share': _('Users with whom to share shopping lists.'),
|
||||||
'ingredient_decimals': _('Number of decimals to round ingredients.'),
|
'ingredient_decimals': _('Number of decimals to round ingredients.'),
|
||||||
'comments': _('If you want to be able to create and see comments underneath recipes.'),
|
'comments': _('If you want to be able to create and see comments underneath recipes.'),
|
||||||
'shopping_auto_sync': _(
|
'shopping_auto_sync': _(
|
||||||
'Setting to 0 will disable auto sync. When viewing a shopping list the list is updated every set seconds to sync changes someone else might have made. Useful when shopping with multiple people but might use a little bit '
|
'Setting to 0 will disable auto sync. When viewing a shopping list the list is updated every set seconds to sync changes someone else might have made. Useful when shopping with multiple people but might use a little bit '
|
||||||
'of mobile data. If lower than instance limit it is reset when saving.'
|
'of mobile data. If lower than instance limit it is reset when saving.'
|
||||||
),
|
),
|
||||||
'sticky_navbar': _('Makes the navbar stick to the top of the page.'),
|
'sticky_navbar': _('Makes the navbar stick to the top of the page.'),
|
||||||
'mealplan_autoadd_shopping': _('Automatically add meal plan ingredients to shopping list.'),
|
'mealplan_autoadd_shopping': _('Automatically add meal plan ingredients to shopping list.'),
|
||||||
'mealplan_autoexclude_onhand': _('Exclude ingredients that are on hand.'),
|
'mealplan_autoexclude_onhand': _('Exclude ingredients that are on hand.'),
|
||||||
'left_handed': _('Will optimize the UI for use with your left hand.')
|
'left_handed': _('Will optimize the UI for use with your left hand.'),
|
||||||
|
'show_step_ingredients': _('Add ingredients table next to recipe steps. Applies at creation time for manually created and URL imported recipes. Individual steps can be overridden in the edit recipe view.')
|
||||||
}
|
}
|
||||||
|
|
||||||
widgets = {
|
widgets = {
|
||||||
@@ -167,8 +167,26 @@ class ImportExportBase(forms.Form):
|
|||||||
))
|
))
|
||||||
|
|
||||||
|
|
||||||
|
class MultipleFileInput(forms.ClearableFileInput):
|
||||||
|
allow_multiple_selected = True
|
||||||
|
|
||||||
|
|
||||||
|
class MultipleFileField(forms.FileField):
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
kwargs.setdefault("widget", MultipleFileInput())
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def clean(self, data, initial=None):
|
||||||
|
single_file_clean = super().clean
|
||||||
|
if isinstance(data, (list, tuple)):
|
||||||
|
result = [single_file_clean(d, initial) for d in data]
|
||||||
|
else:
|
||||||
|
result = single_file_clean(data, initial)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
class ImportForm(ImportExportBase):
|
class ImportForm(ImportExportBase):
|
||||||
files = forms.FileField(required=True, widget=forms.ClearableFileInput(attrs={'multiple': True}))
|
files = MultipleFileField(required=True)
|
||||||
duplicates = forms.BooleanField(help_text=_(
|
duplicates = forms.BooleanField(help_text=_(
|
||||||
'To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.'),
|
'To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.'),
|
||||||
required=False)
|
required=False)
|
||||||
@@ -305,50 +323,6 @@ class ImportRecipeForm(forms.ModelForm):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
# TODO deprecate
|
|
||||||
class MealPlanForm(forms.ModelForm):
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
space = kwargs.pop('space')
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.fields['recipe'].queryset = Recipe.objects.filter(space=space).all()
|
|
||||||
self.fields['meal_type'].queryset = MealType.objects.filter(space=space).all()
|
|
||||||
self.fields['shared'].queryset = User.objects.filter(userpreference__space=space).all()
|
|
||||||
|
|
||||||
def clean(self):
|
|
||||||
cleaned_data = super(MealPlanForm, self).clean()
|
|
||||||
|
|
||||||
if cleaned_data['title'] == '' and cleaned_data['recipe'] is None:
|
|
||||||
raise forms.ValidationError(
|
|
||||||
_('You must provide at least a recipe or a title.')
|
|
||||||
)
|
|
||||||
|
|
||||||
return cleaned_data
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = MealPlan
|
|
||||||
fields = (
|
|
||||||
'recipe', 'title', 'meal_type', 'note',
|
|
||||||
'servings', 'date', 'shared'
|
|
||||||
)
|
|
||||||
|
|
||||||
help_texts = {
|
|
||||||
'shared': _('You can list default users to share recipes with in the settings.'),
|
|
||||||
'note': _('You can use markdown to format this field. See the <a href="/docs/markdown/">docs here</a>')
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
widgets = {
|
|
||||||
'recipe': SelectWidget,
|
|
||||||
'date': DateWidget,
|
|
||||||
'shared': MultiSelectWidget
|
|
||||||
}
|
|
||||||
field_classes = {
|
|
||||||
'recipe': SafeModelChoiceField,
|
|
||||||
'meal_type': SafeModelChoiceField,
|
|
||||||
'shared': SafeModelMultipleChoiceField,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class InviteLinkForm(forms.ModelForm):
|
class InviteLinkForm(forms.ModelForm):
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
user = kwargs.pop('user')
|
user = kwargs.pop('user')
|
||||||
@@ -489,8 +463,8 @@ class ShoppingPreferenceForm(forms.ModelForm):
|
|||||||
help_texts = {
|
help_texts = {
|
||||||
'shopping_share': _('Users will see all items you add to your shopping list. They must add you to see items on their list.'),
|
'shopping_share': _('Users will see all items you add to your shopping list. They must add you to see items on their list.'),
|
||||||
'shopping_auto_sync': _(
|
'shopping_auto_sync': _(
|
||||||
'Setting to 0 will disable auto sync. When viewing a shopping list the list is updated every set seconds to sync changes someone else might have made. Useful when shopping with multiple people but might use a little bit '
|
'Setting to 0 will disable auto sync. When viewing a shopping list the list is updated every set seconds to sync changes someone else might have made. Useful when shopping with multiple people but might use a little bit '
|
||||||
'of mobile data. If lower than instance limit it is reset when saving.'
|
'of mobile data. If lower than instance limit it is reset when saving.'
|
||||||
),
|
),
|
||||||
'mealplan_autoadd_shopping': _('Automatically add meal plan ingredients to shopping list.'),
|
'mealplan_autoadd_shopping': _('Automatically add meal plan ingredients to shopping list.'),
|
||||||
'mealplan_autoinclude_related': _('When adding a meal plan to the shopping list (manually or automatically), include all related recipes.'),
|
'mealplan_autoinclude_related': _('When adding a meal plan to the shopping list (manually or automatically), include all related recipes.'),
|
||||||
@@ -534,11 +508,10 @@ class SpacePreferenceForm(forms.ModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = Space
|
model = Space
|
||||||
|
|
||||||
fields = ('food_inherit', 'reset_food_inherit', 'show_facet_count', 'use_plural')
|
fields = ('food_inherit', 'reset_food_inherit', 'use_plural')
|
||||||
|
|
||||||
help_texts = {
|
help_texts = {
|
||||||
'food_inherit': _('Fields on food that should be inherited by default.'),
|
'food_inherit': _('Fields on food that should be inherited by default.'),
|
||||||
'show_facet_count': _('Show recipe counts on search filters'),
|
|
||||||
'use_plural': _('Use the plural form for units and food inside this space.'),
|
'use_plural': _('Use the plural form for units and food inside this space.'),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,10 @@
|
|||||||
import datetime
|
import datetime
|
||||||
|
from gettext import gettext as _
|
||||||
from django.conf import settings
|
|
||||||
|
|
||||||
from allauth.account.adapter import DefaultAccountAdapter
|
from allauth.account.adapter import DefaultAccountAdapter
|
||||||
|
from django.conf import settings
|
||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.core.cache import caches
|
from django.core.cache import caches
|
||||||
from gettext import gettext as _
|
|
||||||
|
|
||||||
from cookbook.models import InviteLink
|
from cookbook.models import InviteLink
|
||||||
|
|
||||||
@@ -17,10 +16,13 @@ class AllAuthCustomAdapter(DefaultAccountAdapter):
|
|||||||
Whether to allow sign-ups.
|
Whether to allow sign-ups.
|
||||||
"""
|
"""
|
||||||
signup_token = False
|
signup_token = False
|
||||||
if 'signup_token' in request.session and InviteLink.objects.filter(valid_until__gte=datetime.datetime.today(), used_by=None, uuid=request.session['signup_token']).exists():
|
if 'signup_token' in request.session and InviteLink.objects.filter(
|
||||||
|
valid_until__gte=datetime.datetime.today(), used_by=None, uuid=request.session['signup_token']).exists():
|
||||||
signup_token = True
|
signup_token = True
|
||||||
|
|
||||||
if (request.resolver_match.view_name == 'account_signup' or request.resolver_match.view_name == 'socialaccount_signup') and not settings.ENABLE_SIGNUP and not signup_token:
|
if request.resolver_match.view_name == 'account_signup' and not settings.ENABLE_SIGNUP and not signup_token:
|
||||||
|
return False
|
||||||
|
elif request.resolver_match.view_name == 'socialaccount_signup' and len(settings.SOCIAL_PROVIDERS) < 1:
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
return super(AllAuthCustomAdapter, self).is_open_for_signup(request)
|
return super(AllAuthCustomAdapter, self).is_open_for_signup(request)
|
||||||
@@ -33,7 +35,7 @@ class AllAuthCustomAdapter(DefaultAccountAdapter):
|
|||||||
if c == default:
|
if c == default:
|
||||||
try:
|
try:
|
||||||
super(AllAuthCustomAdapter, self).send_mail(template_prefix, email, context)
|
super(AllAuthCustomAdapter, self).send_mail(template_prefix, email, context)
|
||||||
except Exception: # dont fail signup just because confirmation mail could not be send
|
except Exception: # dont fail signup just because confirmation mail could not be send
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
messages.add_message(self.request, messages.ERROR, _('In order to prevent spam, the requested email was not send. Please wait a few minutes and try again.'))
|
messages.add_message(self.request, messages.ERROR, _('In order to prevent spam, the requested email was not send. Please wait a few minutes and try again.'))
|
||||||
|
|||||||
@@ -1,6 +1,3 @@
|
|||||||
import cookbook.helper.dal
|
|
||||||
from cookbook.helper.AllAuthCustomAdapter import AllAuthCustomAdapter
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
'dal',
|
'dal',
|
||||||
]
|
]
|
||||||
|
|||||||
227
cookbook/helper/automation_helper.py
Normal file
227
cookbook/helper/automation_helper.py
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
import re
|
||||||
|
|
||||||
|
from django.core.cache import caches
|
||||||
|
from django.db.models.functions import Lower
|
||||||
|
|
||||||
|
from cookbook.models import Automation
|
||||||
|
|
||||||
|
|
||||||
|
class AutomationEngine:
|
||||||
|
request = None
|
||||||
|
source = None
|
||||||
|
use_cache = None
|
||||||
|
food_aliases = None
|
||||||
|
keyword_aliases = None
|
||||||
|
unit_aliases = None
|
||||||
|
never_unit = None
|
||||||
|
transpose_words = None
|
||||||
|
regex_replace = {
|
||||||
|
Automation.DESCRIPTION_REPLACE: None,
|
||||||
|
Automation.INSTRUCTION_REPLACE: None,
|
||||||
|
Automation.FOOD_REPLACE: None,
|
||||||
|
Automation.UNIT_REPLACE: None,
|
||||||
|
Automation.NAME_REPLACE: None,
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self, request, use_cache=True, source=None):
|
||||||
|
self.request = request
|
||||||
|
self.use_cache = use_cache
|
||||||
|
if not source:
|
||||||
|
self.source = "default_string_to_avoid_false_regex_match"
|
||||||
|
else:
|
||||||
|
self.source = source
|
||||||
|
|
||||||
|
def apply_keyword_automation(self, keyword):
|
||||||
|
keyword = keyword.strip()
|
||||||
|
if self.use_cache and self.keyword_aliases is None:
|
||||||
|
self.keyword_aliases = {}
|
||||||
|
KEYWORD_CACHE_KEY = f'automation_keyword_alias_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(KEYWORD_CACHE_KEY, None):
|
||||||
|
self.keyword_aliases = c
|
||||||
|
caches['default'].touch(KEYWORD_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.KEYWORD_ALIAS).only('param_1', 'param_2').order_by('order').all():
|
||||||
|
self.keyword_aliases[a.param_1.lower()] = a.param_2
|
||||||
|
caches['default'].set(KEYWORD_CACHE_KEY, self.keyword_aliases, 30)
|
||||||
|
else:
|
||||||
|
self.keyword_aliases = {}
|
||||||
|
if self.keyword_aliases:
|
||||||
|
try:
|
||||||
|
keyword = self.keyword_aliases[keyword.lower()]
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
if automation := Automation.objects.filter(space=self.request.space, type=Automation.KEYWORD_ALIAS, param_1__iexact=keyword, disabled=False).order_by('order').first():
|
||||||
|
return automation.param_2
|
||||||
|
return keyword
|
||||||
|
|
||||||
|
def apply_unit_automation(self, unit):
|
||||||
|
unit = unit.strip()
|
||||||
|
if self.use_cache and self.unit_aliases is None:
|
||||||
|
self.unit_aliases = {}
|
||||||
|
UNIT_CACHE_KEY = f'automation_unit_alias_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(UNIT_CACHE_KEY, None):
|
||||||
|
self.unit_aliases = c
|
||||||
|
caches['default'].touch(UNIT_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.UNIT_ALIAS).only('param_1', 'param_2').order_by('order').all():
|
||||||
|
self.unit_aliases[a.param_1.lower()] = a.param_2
|
||||||
|
caches['default'].set(UNIT_CACHE_KEY, self.unit_aliases, 30)
|
||||||
|
else:
|
||||||
|
self.unit_aliases = {}
|
||||||
|
if self.unit_aliases:
|
||||||
|
try:
|
||||||
|
unit = self.unit_aliases[unit.lower()]
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
if automation := Automation.objects.filter(space=self.request.space, type=Automation.UNIT_ALIAS, param_1__iexact=unit, disabled=False).order_by('order').first():
|
||||||
|
return automation.param_2
|
||||||
|
return self.apply_regex_replace_automation(unit, Automation.UNIT_REPLACE)
|
||||||
|
|
||||||
|
def apply_food_automation(self, food):
|
||||||
|
food = food.strip()
|
||||||
|
if self.use_cache and self.food_aliases is None:
|
||||||
|
self.food_aliases = {}
|
||||||
|
FOOD_CACHE_KEY = f'automation_food_alias_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(FOOD_CACHE_KEY, None):
|
||||||
|
self.food_aliases = c
|
||||||
|
caches['default'].touch(FOOD_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.FOOD_ALIAS).only('param_1', 'param_2').order_by('order').all():
|
||||||
|
self.food_aliases[a.param_1.lower()] = a.param_2
|
||||||
|
caches['default'].set(FOOD_CACHE_KEY, self.food_aliases, 30)
|
||||||
|
else:
|
||||||
|
self.food_aliases = {}
|
||||||
|
|
||||||
|
if self.food_aliases:
|
||||||
|
try:
|
||||||
|
return self.food_aliases[food.lower()]
|
||||||
|
except KeyError:
|
||||||
|
return food
|
||||||
|
else:
|
||||||
|
if automation := Automation.objects.filter(space=self.request.space, type=Automation.FOOD_ALIAS, param_1__iexact=food, disabled=False).order_by('order').first():
|
||||||
|
return automation.param_2
|
||||||
|
return self.apply_regex_replace_automation(food, Automation.FOOD_REPLACE)
|
||||||
|
|
||||||
|
def apply_never_unit_automation(self, tokens):
|
||||||
|
"""
|
||||||
|
Moves a string that should never be treated as a unit to next token and optionally replaced with default unit
|
||||||
|
e.g. NEVER_UNIT: param1: egg, param2: None would modify ['1', 'egg', 'white'] to ['1', '', 'egg', 'white']
|
||||||
|
or NEVER_UNIT: param1: egg, param2: pcs would modify ['1', 'egg', 'yolk'] to ['1', 'pcs', 'egg', 'yolk']
|
||||||
|
:param1 string: string that should never be considered a unit, will be moved to token[2]
|
||||||
|
:param2 (optional) unit as string: will insert unit string into token[1]
|
||||||
|
:return: unit as string (possibly changed by automation)
|
||||||
|
"""
|
||||||
|
|
||||||
|
if self.use_cache and self.never_unit is None:
|
||||||
|
self.never_unit = {}
|
||||||
|
NEVER_UNIT_CACHE_KEY = f'automation_never_unit_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(NEVER_UNIT_CACHE_KEY, None):
|
||||||
|
self.never_unit = c
|
||||||
|
caches['default'].touch(NEVER_UNIT_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.NEVER_UNIT).only('param_1', 'param_2').order_by('order').all():
|
||||||
|
self.never_unit[a.param_1.lower()] = a.param_2
|
||||||
|
caches['default'].set(NEVER_UNIT_CACHE_KEY, self.never_unit, 30)
|
||||||
|
else:
|
||||||
|
self.never_unit = {}
|
||||||
|
|
||||||
|
new_unit = None
|
||||||
|
alt_unit = self.apply_unit_automation(tokens[1])
|
||||||
|
never_unit = False
|
||||||
|
if self.never_unit:
|
||||||
|
try:
|
||||||
|
new_unit = self.never_unit[tokens[1].lower()]
|
||||||
|
never_unit = True
|
||||||
|
except KeyError:
|
||||||
|
return tokens
|
||||||
|
else:
|
||||||
|
if a := Automation.objects.annotate(param_1_lower=Lower('param_1')).filter(space=self.request.space, type=Automation.NEVER_UNIT, param_1_lower__in=[
|
||||||
|
tokens[1].lower(), alt_unit.lower()], disabled=False).order_by('order').first():
|
||||||
|
new_unit = a.param_2
|
||||||
|
never_unit = True
|
||||||
|
|
||||||
|
if never_unit:
|
||||||
|
tokens.insert(1, new_unit)
|
||||||
|
return tokens
|
||||||
|
|
||||||
|
def apply_transpose_automation(self, string):
|
||||||
|
"""
|
||||||
|
If two words (param_1 & param_2) are detected in sequence, swap their position in the ingredient string
|
||||||
|
:param 1: first word to detect
|
||||||
|
:param 2: second word to detect
|
||||||
|
return: new ingredient string
|
||||||
|
"""
|
||||||
|
if self.use_cache and self.transpose_words is None:
|
||||||
|
self.transpose_words = {}
|
||||||
|
TRANSPOSE_WORDS_CACHE_KEY = f'automation_transpose_words_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(TRANSPOSE_WORDS_CACHE_KEY, None):
|
||||||
|
self.transpose_words = c
|
||||||
|
caches['default'].touch(TRANSPOSE_WORDS_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
i = 0
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.TRANSPOSE_WORDS).only(
|
||||||
|
'param_1', 'param_2').order_by('order').all()[:512]:
|
||||||
|
self.transpose_words[i] = [a.param_1.lower(), a.param_2.lower()]
|
||||||
|
i += 1
|
||||||
|
caches['default'].set(TRANSPOSE_WORDS_CACHE_KEY, self.transpose_words, 30)
|
||||||
|
else:
|
||||||
|
self.transpose_words = {}
|
||||||
|
|
||||||
|
tokens = [x.lower() for x in string.replace(',', ' ').split()]
|
||||||
|
if self.transpose_words:
|
||||||
|
for key, value in self.transpose_words.items():
|
||||||
|
if value[0] in tokens and value[1] in tokens:
|
||||||
|
string = re.sub(rf"\b({value[0]})\W*({value[1]})\b", r"\2 \1", string, flags=re.IGNORECASE)
|
||||||
|
else:
|
||||||
|
for rule in Automation.objects.filter(space=self.request.space, type=Automation.TRANSPOSE_WORDS, disabled=False) \
|
||||||
|
.annotate(param_1_lower=Lower('param_1'), param_2_lower=Lower('param_2')) \
|
||||||
|
.filter(param_1_lower__in=tokens, param_2_lower__in=tokens).order_by('order')[:512]:
|
||||||
|
if rule.param_1 in tokens and rule.param_2 in tokens:
|
||||||
|
string = re.sub(rf"\b({rule.param_1})\W*({rule.param_2})\b", r"\2 \1", string, flags=re.IGNORECASE)
|
||||||
|
return string
|
||||||
|
|
||||||
|
def apply_regex_replace_automation(self, string, automation_type):
|
||||||
|
# TODO add warning - maybe on SPACE page? when a max of 512 automations of a specific type is exceeded (ALIAS types excluded?)
|
||||||
|
"""
|
||||||
|
Replaces strings in a recipe field that are from a matched source
|
||||||
|
field_type are Automation.type that apply regex replacements
|
||||||
|
Automation.DESCRIPTION_REPLACE
|
||||||
|
Automation.INSTRUCTION_REPLACE
|
||||||
|
Automation.FOOD_REPLACE
|
||||||
|
Automation.UNIT_REPLACE
|
||||||
|
Automation.NAME_REPLACE
|
||||||
|
|
||||||
|
regex replacment utilized the following fields from the Automation model
|
||||||
|
:param 1: source that should apply the automation in regex format ('.*' for all)
|
||||||
|
:param 2: regex pattern to match ()
|
||||||
|
:param 3: replacement string (leave blank to delete)
|
||||||
|
return: new string
|
||||||
|
"""
|
||||||
|
if self.use_cache and self.regex_replace[automation_type] is None:
|
||||||
|
self.regex_replace[automation_type] = {}
|
||||||
|
REGEX_REPLACE_CACHE_KEY = f'automation_regex_replace_{self.request.space.pk}'
|
||||||
|
if c := caches['default'].get(REGEX_REPLACE_CACHE_KEY, None):
|
||||||
|
self.regex_replace[automation_type] = c[automation_type]
|
||||||
|
caches['default'].touch(REGEX_REPLACE_CACHE_KEY, 30)
|
||||||
|
else:
|
||||||
|
i = 0
|
||||||
|
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=automation_type).only(
|
||||||
|
'param_1', 'param_2', 'param_3').order_by('order').all()[:512]:
|
||||||
|
self.regex_replace[automation_type][i] = [a.param_1, a.param_2, a.param_3]
|
||||||
|
i += 1
|
||||||
|
caches['default'].set(REGEX_REPLACE_CACHE_KEY, self.regex_replace, 30)
|
||||||
|
else:
|
||||||
|
self.regex_replace[automation_type] = {}
|
||||||
|
|
||||||
|
if self.regex_replace[automation_type]:
|
||||||
|
for rule in self.regex_replace[automation_type].values():
|
||||||
|
if re.match(rule[0], (self.source)[:512]):
|
||||||
|
string = re.sub(rule[1], rule[2], string, flags=re.IGNORECASE)
|
||||||
|
else:
|
||||||
|
for rule in Automation.objects.filter(space=self.request.space, disabled=False, type=automation_type).only(
|
||||||
|
'param_1', 'param_2', 'param_3').order_by('order').all()[:512]:
|
||||||
|
if re.match(rule.param_1, (self.source)[:512]):
|
||||||
|
string = re.sub(rule.param_2, rule.param_3, string, flags=re.IGNORECASE)
|
||||||
|
return string
|
||||||
11
cookbook/helper/cache_helper.py
Normal file
11
cookbook/helper/cache_helper.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
class CacheHelper:
|
||||||
|
space = None
|
||||||
|
|
||||||
|
BASE_UNITS_CACHE_KEY = None
|
||||||
|
PROPERTY_TYPE_CACHE_KEY = None
|
||||||
|
|
||||||
|
def __init__(self, space):
|
||||||
|
self.space = space
|
||||||
|
|
||||||
|
self.BASE_UNITS_CACHE_KEY = f'SPACE_{space.id}_BASE_UNITS'
|
||||||
|
self.PROPERTY_TYPE_CACHE_KEY = f'SPACE_{space.id}_PROPERTY_TYPES'
|
||||||
@@ -1,8 +1,7 @@
|
|||||||
import os
|
import os
|
||||||
import sys
|
from io import BytesIO
|
||||||
|
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
from io import BytesIO
|
|
||||||
|
|
||||||
|
|
||||||
def rescale_image_jpeg(image_object, base_width=1020):
|
def rescale_image_jpeg(image_object, base_width=1020):
|
||||||
@@ -40,7 +39,12 @@ def get_filetype(name):
|
|||||||
# TODO also add env variable to define which images sizes should be compressed
|
# TODO also add env variable to define which images sizes should be compressed
|
||||||
# filetype argument can not be optional, otherwise this function will treat all images as if they were a jpeg
|
# filetype argument can not be optional, otherwise this function will treat all images as if they were a jpeg
|
||||||
# Because it's no longer optional, no reason to return it
|
# Because it's no longer optional, no reason to return it
|
||||||
def handle_image(request, image_object, filetype):
|
def handle_image(request, image_object, filetype):
|
||||||
|
try:
|
||||||
|
Image.open(image_object).verify()
|
||||||
|
except Exception:
|
||||||
|
return None
|
||||||
|
|
||||||
if (image_object.size / 1000) > 500: # if larger than 500 kb compress
|
if (image_object.size / 1000) > 500: # if larger than 500 kb compress
|
||||||
if filetype == '.jpeg' or filetype == '.jpg':
|
if filetype == '.jpeg' or filetype == '.jpg':
|
||||||
return rescale_image_jpeg(image_object)
|
return rescale_image_jpeg(image_object)
|
||||||
|
|||||||
@@ -2,18 +2,16 @@ import re
|
|||||||
import string
|
import string
|
||||||
import unicodedata
|
import unicodedata
|
||||||
|
|
||||||
from django.core.cache import caches
|
from cookbook.helper.automation_helper import AutomationEngine
|
||||||
|
from cookbook.models import Food, Ingredient, Unit
|
||||||
from cookbook.models import Unit, Food, Automation, Ingredient
|
|
||||||
|
|
||||||
|
|
||||||
class IngredientParser:
|
class IngredientParser:
|
||||||
request = None
|
request = None
|
||||||
ignore_rules = False
|
ignore_rules = False
|
||||||
food_aliases = {}
|
automation = None
|
||||||
unit_aliases = {}
|
|
||||||
|
|
||||||
def __init__(self, request, cache_mode, ignore_automations=False):
|
def __init__(self, request, cache_mode=True, ignore_automations=False):
|
||||||
"""
|
"""
|
||||||
Initialize ingredient parser
|
Initialize ingredient parser
|
||||||
:param request: request context (to control caching, rule ownership, etc.)
|
:param request: request context (to control caching, rule ownership, etc.)
|
||||||
@@ -22,65 +20,8 @@ class IngredientParser:
|
|||||||
"""
|
"""
|
||||||
self.request = request
|
self.request = request
|
||||||
self.ignore_rules = ignore_automations
|
self.ignore_rules = ignore_automations
|
||||||
if cache_mode:
|
if not self.ignore_rules:
|
||||||
FOOD_CACHE_KEY = f'automation_food_alias_{self.request.space.pk}'
|
self.automation = AutomationEngine(self.request, use_cache=cache_mode)
|
||||||
if c := caches['default'].get(FOOD_CACHE_KEY, None):
|
|
||||||
self.food_aliases = c
|
|
||||||
caches['default'].touch(FOOD_CACHE_KEY, 30)
|
|
||||||
else:
|
|
||||||
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.FOOD_ALIAS).only('param_1', 'param_2').order_by('order').all():
|
|
||||||
self.food_aliases[a.param_1] = a.param_2
|
|
||||||
caches['default'].set(FOOD_CACHE_KEY, self.food_aliases, 30)
|
|
||||||
|
|
||||||
UNIT_CACHE_KEY = f'automation_unit_alias_{self.request.space.pk}'
|
|
||||||
if c := caches['default'].get(UNIT_CACHE_KEY, None):
|
|
||||||
self.unit_aliases = c
|
|
||||||
caches['default'].touch(UNIT_CACHE_KEY, 30)
|
|
||||||
else:
|
|
||||||
for a in Automation.objects.filter(space=self.request.space, disabled=False, type=Automation.UNIT_ALIAS).only('param_1', 'param_2').order_by('order').all():
|
|
||||||
self.unit_aliases[a.param_1] = a.param_2
|
|
||||||
caches['default'].set(UNIT_CACHE_KEY, self.unit_aliases, 30)
|
|
||||||
else:
|
|
||||||
self.food_aliases = {}
|
|
||||||
self.unit_aliases = {}
|
|
||||||
|
|
||||||
def apply_food_automation(self, food):
|
|
||||||
"""
|
|
||||||
Apply food alias automations to passed food
|
|
||||||
:param food: unit as string
|
|
||||||
:return: food as string (possibly changed by automation)
|
|
||||||
"""
|
|
||||||
if self.ignore_rules:
|
|
||||||
return food
|
|
||||||
else:
|
|
||||||
if self.food_aliases:
|
|
||||||
try:
|
|
||||||
return self.food_aliases[food]
|
|
||||||
except KeyError:
|
|
||||||
return food
|
|
||||||
else:
|
|
||||||
if automation := Automation.objects.filter(space=self.request.space, type=Automation.FOOD_ALIAS, param_1=food, disabled=False).order_by('order').first():
|
|
||||||
return automation.param_2
|
|
||||||
return food
|
|
||||||
|
|
||||||
def apply_unit_automation(self, unit):
|
|
||||||
"""
|
|
||||||
Apply unit alias automations to passed unit
|
|
||||||
:param unit: unit as string
|
|
||||||
:return: unit as string (possibly changed by automation)
|
|
||||||
"""
|
|
||||||
if self.ignore_rules:
|
|
||||||
return unit
|
|
||||||
else:
|
|
||||||
if self.unit_aliases:
|
|
||||||
try:
|
|
||||||
return self.unit_aliases[unit]
|
|
||||||
except KeyError:
|
|
||||||
return unit
|
|
||||||
else:
|
|
||||||
if automation := Automation.objects.filter(space=self.request.space, type=Automation.UNIT_ALIAS, param_1=unit, disabled=False).order_by('order').first():
|
|
||||||
return automation.param_2
|
|
||||||
return unit
|
|
||||||
|
|
||||||
def get_unit(self, unit):
|
def get_unit(self, unit):
|
||||||
"""
|
"""
|
||||||
@@ -91,7 +32,10 @@ class IngredientParser:
|
|||||||
if not unit:
|
if not unit:
|
||||||
return None
|
return None
|
||||||
if len(unit) > 0:
|
if len(unit) > 0:
|
||||||
u, created = Unit.objects.get_or_create(name=self.apply_unit_automation(unit), space=self.request.space)
|
if self.ignore_rules:
|
||||||
|
u, created = Unit.objects.get_or_create(name=unit.strip(), space=self.request.space)
|
||||||
|
else:
|
||||||
|
u, created = Unit.objects.get_or_create(name=self.automation.apply_unit_automation(unit), space=self.request.space)
|
||||||
return u
|
return u
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@@ -104,7 +48,10 @@ class IngredientParser:
|
|||||||
if not food:
|
if not food:
|
||||||
return None
|
return None
|
||||||
if len(food) > 0:
|
if len(food) > 0:
|
||||||
f, created = Food.objects.get_or_create(name=self.apply_food_automation(food), space=self.request.space)
|
if self.ignore_rules:
|
||||||
|
f, created = Food.objects.get_or_create(name=food.strip(), space=self.request.space)
|
||||||
|
else:
|
||||||
|
f, created = Food.objects.get_or_create(name=self.automation.apply_food_automation(food), space=self.request.space)
|
||||||
return f
|
return f
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@@ -133,10 +80,10 @@ class IngredientParser:
|
|||||||
end = 0
|
end = 0
|
||||||
while (end < len(x) and (x[end] in string.digits
|
while (end < len(x) and (x[end] in string.digits
|
||||||
or (
|
or (
|
||||||
(x[end] == '.' or x[end] == ',' or x[end] == '/')
|
(x[end] == '.' or x[end] == ',' or x[end] == '/')
|
||||||
and end + 1 < len(x)
|
and end + 1 < len(x)
|
||||||
and x[end + 1] in string.digits
|
and x[end + 1] in string.digits
|
||||||
))):
|
))):
|
||||||
end += 1
|
end += 1
|
||||||
if end > 0:
|
if end > 0:
|
||||||
if "/" in x[:end]:
|
if "/" in x[:end]:
|
||||||
@@ -160,7 +107,8 @@ class IngredientParser:
|
|||||||
if unit is not None and unit.strip() == '':
|
if unit is not None and unit.strip() == '':
|
||||||
unit = None
|
unit = None
|
||||||
|
|
||||||
if unit is not None and (unit.startswith('(') or unit.startswith('-')): # i dont know any unit that starts with ( or - so its likely an alternative like 1L (500ml) Water or 2-3
|
if unit is not None and (unit.startswith('(') or unit.startswith(
|
||||||
|
'-')): # i dont know any unit that starts with ( or - so its likely an alternative like 1L (500ml) Water or 2-3
|
||||||
unit = None
|
unit = None
|
||||||
note = x
|
note = x
|
||||||
return amount, unit, note
|
return amount, unit, note
|
||||||
@@ -230,8 +178,8 @@ class IngredientParser:
|
|||||||
|
|
||||||
# if the string contains parenthesis early on remove it and place it at the end
|
# if the string contains parenthesis early on remove it and place it at the end
|
||||||
# because its likely some kind of note
|
# because its likely some kind of note
|
||||||
if re.match('(.){1,6}\s\((.[^\(\)])+\)\s', ingredient):
|
if re.match('(.){1,6}\\s\\((.[^\\(\\)])+\\)\\s', ingredient):
|
||||||
match = re.search('\((.[^\(])+\)', ingredient)
|
match = re.search('\\((.[^\\(])+\\)', ingredient)
|
||||||
ingredient = ingredient[:match.start()] + ingredient[match.end():] + ' ' + ingredient[match.start():match.end()]
|
ingredient = ingredient[:match.start()] + ingredient[match.end():] + ' ' + ingredient[match.start():match.end()]
|
||||||
|
|
||||||
# leading spaces before commas result in extra tokens, clean them out
|
# leading spaces before commas result in extra tokens, clean them out
|
||||||
@@ -239,12 +187,15 @@ class IngredientParser:
|
|||||||
|
|
||||||
# handle "(from) - (to)" amounts by using the minimum amount and adding the range to the description
|
# handle "(from) - (to)" amounts by using the minimum amount and adding the range to the description
|
||||||
# "10.5 - 200 g XYZ" => "100 g XYZ (10.5 - 200)"
|
# "10.5 - 200 g XYZ" => "100 g XYZ (10.5 - 200)"
|
||||||
ingredient = re.sub("^(\d+|\d+[\\.,]\d+) - (\d+|\d+[\\.,]\d+) (.*)", "\\1 \\3 (\\1 - \\2)", ingredient)
|
ingredient = re.sub("^(\\d+|\\d+[\\.,]\\d+) - (\\d+|\\d+[\\.,]\\d+) (.*)", "\\1 \\3 (\\1 - \\2)", ingredient)
|
||||||
|
|
||||||
# if amount and unit are connected add space in between
|
# if amount and unit are connected add space in between
|
||||||
if re.match('([0-9])+([A-z])+\s', ingredient):
|
if re.match('([0-9])+([A-z])+\\s', ingredient):
|
||||||
ingredient = re.sub(r'(?<=([a-z])|\d)(?=(?(1)\d|[a-z]))', ' ', ingredient)
|
ingredient = re.sub(r'(?<=([a-z])|\d)(?=(?(1)\d|[a-z]))', ' ', ingredient)
|
||||||
|
|
||||||
|
if not self.ignore_rules:
|
||||||
|
ingredient = self.automation.apply_transpose_automation(ingredient)
|
||||||
|
|
||||||
tokens = ingredient.split() # split at each space into tokens
|
tokens = ingredient.split() # split at each space into tokens
|
||||||
if len(tokens) == 1:
|
if len(tokens) == 1:
|
||||||
# there only is one argument, that must be the food
|
# there only is one argument, that must be the food
|
||||||
@@ -257,6 +208,8 @@ class IngredientParser:
|
|||||||
# three arguments if it already has a unit there can't be
|
# three arguments if it already has a unit there can't be
|
||||||
# a fraction for the amount
|
# a fraction for the amount
|
||||||
if len(tokens) > 2:
|
if len(tokens) > 2:
|
||||||
|
if not self.ignore_rules:
|
||||||
|
tokens = self.automation.apply_never_unit_automation(tokens)
|
||||||
try:
|
try:
|
||||||
if unit is not None:
|
if unit is not None:
|
||||||
# a unit is already found, no need to try the second argument for a fraction
|
# a unit is already found, no need to try the second argument for a fraction
|
||||||
@@ -303,10 +256,11 @@ class IngredientParser:
|
|||||||
if unit_note not in note:
|
if unit_note not in note:
|
||||||
note += ' ' + unit_note
|
note += ' ' + unit_note
|
||||||
|
|
||||||
if unit:
|
if unit and not self.ignore_rules:
|
||||||
unit = self.apply_unit_automation(unit.strip())
|
unit = self.automation.apply_unit_automation(unit)
|
||||||
|
|
||||||
food = self.apply_food_automation(food.strip())
|
if food and not self.ignore_rules:
|
||||||
|
food = self.automation.apply_food_automation(food)
|
||||||
if len(food) > Food._meta.get_field('name').max_length: # test if food name is to long
|
if len(food) > Food._meta.get_field('name').max_length: # test if food name is to long
|
||||||
# try splitting it at a space and taking only the first arg
|
# try splitting it at a space and taking only the first arg
|
||||||
if len(food.split()) > 1 and len(food.split()[0]) < Food._meta.get_field('name').max_length:
|
if len(food.split()) > 1 and len(food.split()[0]) < Food._meta.get_field('name').max_length:
|
||||||
|
|||||||
204
cookbook/helper/open_data_importer.py
Normal file
204
cookbook/helper/open_data_importer.py
Normal file
@@ -0,0 +1,204 @@
|
|||||||
|
from cookbook.models import (Food, FoodProperty, Property, PropertyType, Supermarket,
|
||||||
|
SupermarketCategory, SupermarketCategoryRelation, Unit, UnitConversion)
|
||||||
|
|
||||||
|
|
||||||
|
class OpenDataImporter:
|
||||||
|
request = None
|
||||||
|
data = {}
|
||||||
|
slug_id_cache = {}
|
||||||
|
update_existing = False
|
||||||
|
use_metric = True
|
||||||
|
|
||||||
|
def __init__(self, request, data, update_existing=False, use_metric=True):
|
||||||
|
self.request = request
|
||||||
|
self.data = data
|
||||||
|
self.update_existing = update_existing
|
||||||
|
self.use_metric = use_metric
|
||||||
|
|
||||||
|
def _update_slug_cache(self, object_class, datatype):
|
||||||
|
self.slug_id_cache[datatype] = dict(object_class.objects.filter(space=self.request.space, open_data_slug__isnull=False).values_list('open_data_slug', 'id', ))
|
||||||
|
|
||||||
|
def import_units(self):
|
||||||
|
datatype = 'unit'
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
for u in list(self.data[datatype].keys()):
|
||||||
|
insert_list.append(Unit(
|
||||||
|
name=self.data[datatype][u]['name'],
|
||||||
|
plural_name=self.data[datatype][u]['plural_name'],
|
||||||
|
base_unit=self.data[datatype][u]['base_unit'] if self.data[datatype][u]['base_unit'] != '' else None,
|
||||||
|
open_data_slug=u,
|
||||||
|
space=self.request.space
|
||||||
|
))
|
||||||
|
|
||||||
|
if self.update_existing:
|
||||||
|
return Unit.objects.bulk_create(insert_list, update_conflicts=True, update_fields=(
|
||||||
|
'name', 'plural_name', 'base_unit', 'open_data_slug'), unique_fields=('space', 'name',))
|
||||||
|
else:
|
||||||
|
return Unit.objects.bulk_create(insert_list, update_conflicts=True, update_fields=('open_data_slug',), unique_fields=('space', 'name',))
|
||||||
|
|
||||||
|
def import_category(self):
|
||||||
|
datatype = 'category'
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
insert_list.append(SupermarketCategory(
|
||||||
|
name=self.data[datatype][k]['name'],
|
||||||
|
open_data_slug=k,
|
||||||
|
space=self.request.space
|
||||||
|
))
|
||||||
|
|
||||||
|
return SupermarketCategory.objects.bulk_create(insert_list, update_conflicts=True, update_fields=('open_data_slug',), unique_fields=('space', 'name',))
|
||||||
|
|
||||||
|
def import_property(self):
|
||||||
|
datatype = 'property'
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
insert_list.append(PropertyType(
|
||||||
|
name=self.data[datatype][k]['name'],
|
||||||
|
unit=self.data[datatype][k]['unit'],
|
||||||
|
open_data_slug=k,
|
||||||
|
space=self.request.space
|
||||||
|
))
|
||||||
|
|
||||||
|
return PropertyType.objects.bulk_create(insert_list, update_conflicts=True, update_fields=('open_data_slug',), unique_fields=('space', 'name',))
|
||||||
|
|
||||||
|
def import_supermarket(self):
|
||||||
|
datatype = 'store'
|
||||||
|
|
||||||
|
self._update_slug_cache(SupermarketCategory, 'category')
|
||||||
|
insert_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
insert_list.append(Supermarket(
|
||||||
|
name=self.data[datatype][k]['name'],
|
||||||
|
open_data_slug=k,
|
||||||
|
space=self.request.space
|
||||||
|
))
|
||||||
|
|
||||||
|
# always add open data slug if matching supermarket is found, otherwise relation might fail
|
||||||
|
supermarkets = Supermarket.objects.bulk_create(insert_list, unique_fields=('space', 'name',), update_conflicts=True, update_fields=('open_data_slug',))
|
||||||
|
self._update_slug_cache(Supermarket, 'store')
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
relations = []
|
||||||
|
order = 0
|
||||||
|
for c in self.data[datatype][k]['categories']:
|
||||||
|
relations.append(
|
||||||
|
SupermarketCategoryRelation(
|
||||||
|
supermarket_id=self.slug_id_cache[datatype][k],
|
||||||
|
category_id=self.slug_id_cache['category'][c],
|
||||||
|
order=order,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
order += 1
|
||||||
|
|
||||||
|
SupermarketCategoryRelation.objects.bulk_create(relations, ignore_conflicts=True, unique_fields=('supermarket', 'category',))
|
||||||
|
|
||||||
|
return supermarkets
|
||||||
|
|
||||||
|
def import_food(self):
|
||||||
|
identifier_list = []
|
||||||
|
datatype = 'food'
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
identifier_list.append(self.data[datatype][k]['name'])
|
||||||
|
identifier_list.append(self.data[datatype][k]['plural_name'])
|
||||||
|
|
||||||
|
existing_objects_flat = []
|
||||||
|
existing_objects = {}
|
||||||
|
for f in Food.objects.filter(space=self.request.space).filter(name__in=identifier_list).values_list('id', 'name', 'plural_name'):
|
||||||
|
existing_objects_flat.append(f[1])
|
||||||
|
existing_objects_flat.append(f[2])
|
||||||
|
existing_objects[f[1]] = f
|
||||||
|
existing_objects[f[2]] = f
|
||||||
|
|
||||||
|
self._update_slug_cache(Unit, 'unit')
|
||||||
|
self._update_slug_cache(PropertyType, 'property')
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
update_list = []
|
||||||
|
update_field_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
if not (self.data[datatype][k]['name'] in existing_objects_flat or self.data[datatype][k]['plural_name'] in existing_objects_flat):
|
||||||
|
insert_list.append({'data': {
|
||||||
|
'name': self.data[datatype][k]['name'],
|
||||||
|
'plural_name': self.data[datatype][k]['plural_name'] if self.data[datatype][k]['plural_name'] != '' else None,
|
||||||
|
'supermarket_category_id': self.slug_id_cache['category'][self.data[datatype][k]['store_category']],
|
||||||
|
'fdc_id': self.data[datatype][k]['fdc_id'] if self.data[datatype][k]['fdc_id'] != '' else None,
|
||||||
|
'open_data_slug': k,
|
||||||
|
'space': self.request.space.id,
|
||||||
|
}})
|
||||||
|
else:
|
||||||
|
if self.data[datatype][k]['name'] in existing_objects:
|
||||||
|
existing_food_id = existing_objects[self.data[datatype][k]['name']][0]
|
||||||
|
else:
|
||||||
|
existing_food_id = existing_objects[self.data[datatype][k]['plural_name']][0]
|
||||||
|
|
||||||
|
if self.update_existing:
|
||||||
|
update_field_list = ['name', 'plural_name', 'preferred_unit_id', 'preferred_shopping_unit_id', 'supermarket_category_id', 'fdc_id', 'open_data_slug', ]
|
||||||
|
update_list.append(Food(
|
||||||
|
id=existing_food_id,
|
||||||
|
name=self.data[datatype][k]['name'],
|
||||||
|
plural_name=self.data[datatype][k]['plural_name'] if self.data[datatype][k]['plural_name'] != '' else None,
|
||||||
|
supermarket_category_id=self.slug_id_cache['category'][self.data[datatype][k]['store_category']],
|
||||||
|
fdc_id=self.data[datatype][k]['fdc_id'] if self.data[datatype][k]['fdc_id'] != '' else None,
|
||||||
|
open_data_slug=k,
|
||||||
|
))
|
||||||
|
else:
|
||||||
|
update_field_list = ['open_data_slug', ]
|
||||||
|
update_list.append(Food(id=existing_food_id, open_data_slug=k, ))
|
||||||
|
|
||||||
|
Food.load_bulk(insert_list, None)
|
||||||
|
if len(update_list) > 0:
|
||||||
|
Food.objects.bulk_update(update_list, update_field_list)
|
||||||
|
|
||||||
|
self._update_slug_cache(Food, 'food')
|
||||||
|
|
||||||
|
food_property_list = []
|
||||||
|
# alias_list = []
|
||||||
|
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
for fp in self.data[datatype][k]['properties']['type_values']:
|
||||||
|
# try catch here because somettimes key "k" is not set for he food cache
|
||||||
|
try:
|
||||||
|
food_property_list.append(Property(
|
||||||
|
property_type_id=self.slug_id_cache['property'][fp['property_type']],
|
||||||
|
property_amount=fp['property_value'],
|
||||||
|
import_food_id=self.slug_id_cache['food'][k],
|
||||||
|
space=self.request.space,
|
||||||
|
))
|
||||||
|
except KeyError:
|
||||||
|
print(str(k) + ' is not in self.slug_id_cache["food"]')
|
||||||
|
|
||||||
|
Property.objects.bulk_create(food_property_list, ignore_conflicts=True, unique_fields=('space', 'import_food_id', 'property_type',))
|
||||||
|
|
||||||
|
property_food_relation_list = []
|
||||||
|
for p in Property.objects.filter(space=self.request.space, import_food_id__isnull=False).values_list('import_food_id', 'id', ):
|
||||||
|
property_food_relation_list.append(Food.properties.through(food_id=p[0], property_id=p[1]))
|
||||||
|
|
||||||
|
FoodProperty.objects.bulk_create(property_food_relation_list, ignore_conflicts=True, unique_fields=('food_id', 'property_id',))
|
||||||
|
|
||||||
|
return insert_list + update_list
|
||||||
|
|
||||||
|
def import_conversion(self):
|
||||||
|
datatype = 'conversion'
|
||||||
|
|
||||||
|
insert_list = []
|
||||||
|
for k in list(self.data[datatype].keys()):
|
||||||
|
# try catch here because sometimes key "k" is not set for he food cache
|
||||||
|
try:
|
||||||
|
insert_list.append(UnitConversion(
|
||||||
|
base_amount=self.data[datatype][k]['base_amount'],
|
||||||
|
base_unit_id=self.slug_id_cache['unit'][self.data[datatype][k]['base_unit']],
|
||||||
|
converted_amount=self.data[datatype][k]['converted_amount'],
|
||||||
|
converted_unit_id=self.slug_id_cache['unit'][self.data[datatype][k]['converted_unit']],
|
||||||
|
food_id=self.slug_id_cache['food'][self.data[datatype][k]['food']],
|
||||||
|
open_data_slug=k,
|
||||||
|
space=self.request.space,
|
||||||
|
created_by=self.request.user,
|
||||||
|
))
|
||||||
|
except KeyError:
|
||||||
|
print(str(k) + ' is not in self.slug_id_cache["food"]')
|
||||||
|
|
||||||
|
return UnitConversion.objects.bulk_create(insert_list, ignore_conflicts=True, unique_fields=('space', 'base_unit', 'converted_unit', 'food', 'open_data_slug'))
|
||||||
@@ -4,16 +4,16 @@ from django.conf import settings
|
|||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.contrib.auth.decorators import user_passes_test
|
from django.contrib.auth.decorators import user_passes_test
|
||||||
from django.core.cache import cache
|
from django.core.cache import cache
|
||||||
from django.core.exceptions import ValidationError, ObjectDoesNotExist
|
from django.core.exceptions import ObjectDoesNotExist, ValidationError
|
||||||
from django.http import HttpResponseRedirect
|
from django.http import HttpResponseRedirect
|
||||||
from django.urls import reverse, reverse_lazy
|
from django.urls import reverse, reverse_lazy
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
from oauth2_provider.contrib.rest_framework import TokenHasScope, TokenHasReadWriteScope
|
from oauth2_provider.contrib.rest_framework import TokenHasReadWriteScope, TokenHasScope
|
||||||
from oauth2_provider.models import AccessToken
|
from oauth2_provider.models import AccessToken
|
||||||
from rest_framework import permissions
|
from rest_framework import permissions
|
||||||
from rest_framework.permissions import SAFE_METHODS
|
from rest_framework.permissions import SAFE_METHODS
|
||||||
|
|
||||||
from cookbook.models import ShareLink, Recipe, UserSpace
|
from cookbook.models import Recipe, ShareLink, UserSpace
|
||||||
|
|
||||||
|
|
||||||
def get_allowed_groups(groups_required):
|
def get_allowed_groups(groups_required):
|
||||||
@@ -255,9 +255,6 @@ class CustomIsShared(permissions.BasePermission):
|
|||||||
return request.user.is_authenticated
|
return request.user.is_authenticated
|
||||||
|
|
||||||
def has_object_permission(self, request, view, obj):
|
def has_object_permission(self, request, view, obj):
|
||||||
# # temporary hack to make old shopping list work with new shopping list
|
|
||||||
# if obj.__class__.__name__ in ['ShoppingList', 'ShoppingListEntry']:
|
|
||||||
# return is_object_shared(request.user, obj) or obj.created_by in list(request.user.get_shopping_share())
|
|
||||||
return is_object_shared(request.user, obj)
|
return is_object_shared(request.user, obj)
|
||||||
|
|
||||||
|
|
||||||
@@ -322,7 +319,8 @@ class CustomRecipePermission(permissions.BasePermission):
|
|||||||
|
|
||||||
def has_permission(self, request, view): # user is either at least a guest or a share link is given and the request is safe
|
def has_permission(self, request, view): # user is either at least a guest or a share link is given and the request is safe
|
||||||
share = request.query_params.get('share', None)
|
share = request.query_params.get('share', None)
|
||||||
return has_group_permission(request.user, ['guest']) or (share and request.method in SAFE_METHODS and 'pk' in view.kwargs)
|
return ((has_group_permission(request.user, ['guest']) and request.method in SAFE_METHODS) or has_group_permission(
|
||||||
|
request.user, ['user'])) or (share and request.method in SAFE_METHODS and 'pk' in view.kwargs)
|
||||||
|
|
||||||
def has_object_permission(self, request, view, obj):
|
def has_object_permission(self, request, view, obj):
|
||||||
share = request.query_params.get('share', None)
|
share = request.query_params.get('share', None)
|
||||||
@@ -332,7 +330,8 @@ class CustomRecipePermission(permissions.BasePermission):
|
|||||||
if obj.private:
|
if obj.private:
|
||||||
return ((obj.created_by == request.user) or (request.user in obj.shared.all())) and obj.space == request.space
|
return ((obj.created_by == request.user) or (request.user in obj.shared.all())) and obj.space == request.space
|
||||||
else:
|
else:
|
||||||
return has_group_permission(request.user, ['guest']) and obj.space == request.space
|
return ((has_group_permission(request.user, ['guest']) and request.method in SAFE_METHODS)
|
||||||
|
or has_group_permission(request.user, ['user'])) and obj.space == request.space
|
||||||
|
|
||||||
|
|
||||||
class CustomUserPermission(permissions.BasePermission):
|
class CustomUserPermission(permissions.BasePermission):
|
||||||
@@ -361,7 +360,7 @@ class CustomTokenHasScope(TokenHasScope):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def has_permission(self, request, view):
|
def has_permission(self, request, view):
|
||||||
if type(request.auth) == AccessToken:
|
if isinstance(request.auth, AccessToken):
|
||||||
return super().has_permission(request, view)
|
return super().has_permission(request, view)
|
||||||
else:
|
else:
|
||||||
return request.user.is_authenticated
|
return request.user.is_authenticated
|
||||||
@@ -375,7 +374,7 @@ class CustomTokenHasReadWriteScope(TokenHasReadWriteScope):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def has_permission(self, request, view):
|
def has_permission(self, request, view):
|
||||||
if type(request.auth) == AccessToken:
|
if isinstance(request.auth, AccessToken):
|
||||||
return super().has_permission(request, view)
|
return super().has_permission(request, view)
|
||||||
else:
|
else:
|
||||||
return True
|
return True
|
||||||
@@ -434,3 +433,10 @@ def switch_user_active_space(user, space):
|
|||||||
return us
|
return us
|
||||||
except ObjectDoesNotExist:
|
except ObjectDoesNotExist:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class IsReadOnlyDRF(permissions.BasePermission):
|
||||||
|
message = 'You cannot interact with this object as it is not owned by you!'
|
||||||
|
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
return request.method in SAFE_METHODS
|
||||||
|
|||||||
74
cookbook/helper/property_helper.py
Normal file
74
cookbook/helper/property_helper.py
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
from django.core.cache import caches
|
||||||
|
|
||||||
|
from cookbook.helper.cache_helper import CacheHelper
|
||||||
|
from cookbook.helper.unit_conversion_helper import UnitConversionHelper
|
||||||
|
from cookbook.models import PropertyType
|
||||||
|
|
||||||
|
|
||||||
|
class FoodPropertyHelper:
|
||||||
|
space = None
|
||||||
|
|
||||||
|
def __init__(self, space):
|
||||||
|
"""
|
||||||
|
Helper to perform food property calculations
|
||||||
|
:param space: space to limit scope to
|
||||||
|
"""
|
||||||
|
self.space = space
|
||||||
|
|
||||||
|
def calculate_recipe_properties(self, recipe):
|
||||||
|
"""
|
||||||
|
Calculate all food properties for a given recipe.
|
||||||
|
:param recipe: recipe to calculate properties for
|
||||||
|
:return: dict of with property keys and total/food values for each property available
|
||||||
|
"""
|
||||||
|
ingredients = []
|
||||||
|
computed_properties = {}
|
||||||
|
|
||||||
|
for s in recipe.steps.all():
|
||||||
|
ingredients += s.ingredients.all()
|
||||||
|
|
||||||
|
property_types = caches['default'].get(CacheHelper(self.space).PROPERTY_TYPE_CACHE_KEY, None)
|
||||||
|
|
||||||
|
if not property_types:
|
||||||
|
property_types = PropertyType.objects.filter(space=self.space).all()
|
||||||
|
# cache is cleared on property type save signal so long duration is fine
|
||||||
|
caches['default'].set(CacheHelper(self.space).PROPERTY_TYPE_CACHE_KEY, property_types, 60 * 60)
|
||||||
|
|
||||||
|
for fpt in property_types:
|
||||||
|
computed_properties[fpt.id] = {'id': fpt.id, 'name': fpt.name, 'description': fpt.description,
|
||||||
|
'unit': fpt.unit, 'order': fpt.order, 'food_values': {}, 'total_value': 0, 'missing_value': False}
|
||||||
|
|
||||||
|
uch = UnitConversionHelper(self.space)
|
||||||
|
|
||||||
|
for i in ingredients:
|
||||||
|
if i.food is not None:
|
||||||
|
conversions = uch.get_conversions(i)
|
||||||
|
for pt in property_types:
|
||||||
|
found_property = False
|
||||||
|
if i.food.properties_food_amount == 0 or i.food.properties_food_unit is None:
|
||||||
|
computed_properties[pt.id]['food_values'][i.food.id] = {'id': i.food.id, 'food': i.food.name, 'value': 0}
|
||||||
|
computed_properties[pt.id]['missing_value'] = i.food.properties_food_unit is None
|
||||||
|
else:
|
||||||
|
for p in i.food.properties.all():
|
||||||
|
if p.property_type == pt:
|
||||||
|
for c in conversions:
|
||||||
|
if c.unit == i.food.properties_food_unit:
|
||||||
|
found_property = True
|
||||||
|
computed_properties[pt.id]['total_value'] += (c.amount / i.food.properties_food_amount) * p.property_amount
|
||||||
|
computed_properties[pt.id]['food_values'] = self.add_or_create(
|
||||||
|
computed_properties[p.property_type.id]['food_values'], c.food.id, (c.amount / i.food.properties_food_amount) * p.property_amount, c.food)
|
||||||
|
if not found_property:
|
||||||
|
computed_properties[pt.id]['missing_value'] = True
|
||||||
|
computed_properties[pt.id]['food_values'][i.food.id] = {'id': i.food.id, 'food': i.food.name, 'value': 0}
|
||||||
|
|
||||||
|
return computed_properties
|
||||||
|
|
||||||
|
# small dict helper to add to existing key or create new, probably a better way of doing this
|
||||||
|
# TODO move to central helper ?
|
||||||
|
@staticmethod
|
||||||
|
def add_or_create(d, key, value, food):
|
||||||
|
if key in d:
|
||||||
|
d[key]['value'] += value
|
||||||
|
else:
|
||||||
|
d[key] = {'id': food.id, 'food': food.name, 'value': value}
|
||||||
|
return d
|
||||||
@@ -1,191 +0,0 @@
|
|||||||
# import json
|
|
||||||
# import re
|
|
||||||
# from json import JSONDecodeError
|
|
||||||
# from urllib.parse import unquote
|
|
||||||
|
|
||||||
# from bs4 import BeautifulSoup
|
|
||||||
# from bs4.element import Tag
|
|
||||||
# from recipe_scrapers import scrape_html, scrape_me
|
|
||||||
# from recipe_scrapers._exceptions import NoSchemaFoundInWildMode
|
|
||||||
# from recipe_scrapers._utils import get_host_name, normalize_string
|
|
||||||
|
|
||||||
# from cookbook.helper import recipe_url_import as helper
|
|
||||||
# from cookbook.helper.scrapers.scrapers import text_scraper
|
|
||||||
|
|
||||||
|
|
||||||
# def get_recipe_from_source(text, url, request):
|
|
||||||
# def build_node(k, v):
|
|
||||||
# if isinstance(v, dict):
|
|
||||||
# node = {
|
|
||||||
# 'name': k,
|
|
||||||
# 'value': k,
|
|
||||||
# 'children': get_children_dict(v)
|
|
||||||
# }
|
|
||||||
# elif isinstance(v, list):
|
|
||||||
# node = {
|
|
||||||
# 'name': k,
|
|
||||||
# 'value': k,
|
|
||||||
# 'children': get_children_list(v)
|
|
||||||
# }
|
|
||||||
# else:
|
|
||||||
# node = {
|
|
||||||
# 'name': k + ": " + normalize_string(str(v)),
|
|
||||||
# 'value': normalize_string(str(v))
|
|
||||||
# }
|
|
||||||
# return node
|
|
||||||
|
|
||||||
# def get_children_dict(children):
|
|
||||||
# kid_list = []
|
|
||||||
# for k, v in children.items():
|
|
||||||
# kid_list.append(build_node(k, v))
|
|
||||||
# return kid_list
|
|
||||||
|
|
||||||
# def get_children_list(children):
|
|
||||||
# kid_list = []
|
|
||||||
# for kid in children:
|
|
||||||
# if type(kid) == list:
|
|
||||||
# node = {
|
|
||||||
# 'name': "unknown list",
|
|
||||||
# 'value': "unknown list",
|
|
||||||
# 'children': get_children_list(kid)
|
|
||||||
# }
|
|
||||||
# kid_list.append(node)
|
|
||||||
# elif type(kid) == dict:
|
|
||||||
# for k, v in kid.items():
|
|
||||||
# kid_list.append(build_node(k, v))
|
|
||||||
# else:
|
|
||||||
# kid_list.append({
|
|
||||||
# 'name': normalize_string(str(kid)),
|
|
||||||
# 'value': normalize_string(str(kid))
|
|
||||||
# })
|
|
||||||
# return kid_list
|
|
||||||
|
|
||||||
# recipe_tree = []
|
|
||||||
# parse_list = []
|
|
||||||
# soup = BeautifulSoup(text, "html.parser")
|
|
||||||
# html_data = get_from_html(soup)
|
|
||||||
# images = get_images_from_source(soup, url)
|
|
||||||
# text = unquote(text)
|
|
||||||
# scrape = None
|
|
||||||
|
|
||||||
# if url and not text:
|
|
||||||
# try:
|
|
||||||
# scrape = scrape_me(url_path=url, wild_mode=True)
|
|
||||||
# except(NoSchemaFoundInWildMode):
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# if not scrape:
|
|
||||||
# try:
|
|
||||||
# parse_list.append(remove_graph(json.loads(text)))
|
|
||||||
# if not url and 'url' in parse_list[0]:
|
|
||||||
# url = parse_list[0]['url']
|
|
||||||
# scrape = text_scraper("<script type='application/ld+json'>" + text + "</script>", url=url)
|
|
||||||
|
|
||||||
# except JSONDecodeError:
|
|
||||||
# for el in soup.find_all('script', type='application/ld+json'):
|
|
||||||
# el = remove_graph(el)
|
|
||||||
# if not url and 'url' in el:
|
|
||||||
# url = el['url']
|
|
||||||
# if type(el) == list:
|
|
||||||
# for le in el:
|
|
||||||
# parse_list.append(le)
|
|
||||||
# elif type(el) == dict:
|
|
||||||
# parse_list.append(el)
|
|
||||||
# for el in soup.find_all(type='application/json'):
|
|
||||||
# el = remove_graph(el)
|
|
||||||
# if type(el) == list:
|
|
||||||
# for le in el:
|
|
||||||
# parse_list.append(le)
|
|
||||||
# elif type(el) == dict:
|
|
||||||
# parse_list.append(el)
|
|
||||||
# scrape = text_scraper(text, url=url)
|
|
||||||
|
|
||||||
# recipe_json = helper.get_from_scraper(scrape, request)
|
|
||||||
|
|
||||||
# # TODO: DEPRECATE recipe_tree & html_data. first validate it isn't used anywhere
|
|
||||||
# for el in parse_list:
|
|
||||||
# temp_tree = []
|
|
||||||
# if isinstance(el, Tag):
|
|
||||||
# try:
|
|
||||||
# el = json.loads(el.string)
|
|
||||||
# except TypeError:
|
|
||||||
# continue
|
|
||||||
|
|
||||||
# for k, v in el.items():
|
|
||||||
# if isinstance(v, dict):
|
|
||||||
# node = {
|
|
||||||
# 'name': k,
|
|
||||||
# 'value': k,
|
|
||||||
# 'children': get_children_dict(v)
|
|
||||||
# }
|
|
||||||
# elif isinstance(v, list):
|
|
||||||
# node = {
|
|
||||||
# 'name': k,
|
|
||||||
# 'value': k,
|
|
||||||
# 'children': get_children_list(v)
|
|
||||||
# }
|
|
||||||
# else:
|
|
||||||
# node = {
|
|
||||||
# 'name': k + ": " + normalize_string(str(v)),
|
|
||||||
# 'value': normalize_string(str(v))
|
|
||||||
# }
|
|
||||||
# temp_tree.append(node)
|
|
||||||
|
|
||||||
# if '@type' in el and el['@type'] == 'Recipe':
|
|
||||||
# recipe_tree += [{'name': 'ld+json', 'children': temp_tree}]
|
|
||||||
# else:
|
|
||||||
# recipe_tree += [{'name': 'json', 'children': temp_tree}]
|
|
||||||
|
|
||||||
# return recipe_json, recipe_tree, html_data, images
|
|
||||||
|
|
||||||
|
|
||||||
# def get_from_html(soup):
|
|
||||||
# INVISIBLE_ELEMS = ('style', 'script', 'head', 'title')
|
|
||||||
# html = []
|
|
||||||
# for s in soup.strings:
|
|
||||||
# if ((s.parent.name not in INVISIBLE_ELEMS) and (len(s.strip()) > 0)):
|
|
||||||
# html.append(s)
|
|
||||||
# return html
|
|
||||||
|
|
||||||
|
|
||||||
# def get_images_from_source(soup, url):
|
|
||||||
# sources = ['src', 'srcset', 'data-src']
|
|
||||||
# images = []
|
|
||||||
# img_tags = soup.find_all('img')
|
|
||||||
# if url:
|
|
||||||
# site = get_host_name(url)
|
|
||||||
# prot = url.split(':')[0]
|
|
||||||
|
|
||||||
# urls = []
|
|
||||||
# for img in img_tags:
|
|
||||||
# for src in sources:
|
|
||||||
# try:
|
|
||||||
# urls.append(img[src])
|
|
||||||
# except KeyError:
|
|
||||||
# pass
|
|
||||||
|
|
||||||
# for u in urls:
|
|
||||||
# u = u.split('?')[0]
|
|
||||||
# filename = re.search(r'/([\w_-]+[.](jpg|jpeg|gif|png))$', u)
|
|
||||||
# if filename:
|
|
||||||
# if (('http' not in u) and (url)):
|
|
||||||
# # sometimes an image source can be relative
|
|
||||||
# # if it is provide the base url
|
|
||||||
# u = '{}://{}{}'.format(prot, site, u)
|
|
||||||
# if 'http' in u:
|
|
||||||
# images.append(u)
|
|
||||||
# return images
|
|
||||||
|
|
||||||
|
|
||||||
# def remove_graph(el):
|
|
||||||
# # recipes type might be wrapped in @graph type
|
|
||||||
# if isinstance(el, Tag):
|
|
||||||
# try:
|
|
||||||
# el = json.loads(el.string)
|
|
||||||
# if '@graph' in el:
|
|
||||||
# for x in el['@graph']:
|
|
||||||
# if '@type' in x and x['@type'] == 'Recipe':
|
|
||||||
# el = x
|
|
||||||
# except (TypeError, JSONDecodeError):
|
|
||||||
# pass
|
|
||||||
# return el
|
|
||||||
@@ -1,14 +1,11 @@
|
|||||||
import json
|
import json
|
||||||
from collections import Counter
|
|
||||||
from datetime import date, timedelta
|
from datetime import date, timedelta
|
||||||
|
|
||||||
from django.contrib.postgres.search import SearchQuery, SearchRank, SearchVector, TrigramSimilarity
|
from django.contrib.postgres.search import SearchQuery, SearchRank, SearchVector, TrigramSimilarity
|
||||||
from django.core.cache import cache
|
from django.core.cache import cache
|
||||||
from django.core.cache import caches
|
from django.db.models import Avg, Case, Count, Exists, F, Max, OuterRef, Q, Subquery, Value, When
|
||||||
from django.db.models import (Avg, Case, Count, Exists, F, Func, Max, OuterRef, Q, Subquery, Value, When, FilteredRelation)
|
|
||||||
from django.db.models.functions import Coalesce, Lower, Substr
|
from django.db.models.functions import Coalesce, Lower, Substr
|
||||||
from django.utils import timezone, translation
|
from django.utils import timezone, translation
|
||||||
from django.utils.translation import gettext as _
|
|
||||||
|
|
||||||
from cookbook.helper.HelperFunctions import Round, str2bool
|
from cookbook.helper.HelperFunctions import Round, str2bool
|
||||||
from cookbook.managers import DICTIONARY
|
from cookbook.managers import DICTIONARY
|
||||||
@@ -17,7 +14,6 @@ from cookbook.models import (CookLog, CustomFilter, Food, Keyword, Recipe, Searc
|
|||||||
from recipes import settings
|
from recipes import settings
|
||||||
|
|
||||||
|
|
||||||
# TODO create extensive tests to make sure ORs ANDs and various filters, sorting, etc work as expected
|
|
||||||
# TODO consider creating a simpleListRecipe API that only includes minimum of recipe info and minimal filtering
|
# TODO consider creating a simpleListRecipe API that only includes minimum of recipe info and minimal filtering
|
||||||
class RecipeSearch():
|
class RecipeSearch():
|
||||||
_postgres = settings.DATABASES['default']['ENGINE'] in ['django.db.backends.postgresql_psycopg2', 'django.db.backends.postgresql']
|
_postgres = settings.DATABASES['default']['ENGINE'] in ['django.db.backends.postgresql_psycopg2', 'django.db.backends.postgresql']
|
||||||
@@ -26,11 +22,17 @@ class RecipeSearch():
|
|||||||
self._request = request
|
self._request = request
|
||||||
self._queryset = None
|
self._queryset = None
|
||||||
if f := params.get('filter', None):
|
if f := params.get('filter', None):
|
||||||
custom_filter = CustomFilter.objects.filter(id=f, space=self._request.space).filter(Q(created_by=self._request.user) |
|
custom_filter = (
|
||||||
Q(shared=self._request.user) | Q(recipebook__shared=self._request.user)).first()
|
CustomFilter.objects.filter(id=f, space=self._request.space)
|
||||||
|
.filter(Q(created_by=self._request.user) | Q(shared=self._request.user) | Q(recipebook__shared=self._request.user))
|
||||||
|
.first()
|
||||||
|
)
|
||||||
if custom_filter:
|
if custom_filter:
|
||||||
self._params = {**json.loads(custom_filter.search)}
|
self._params = {**json.loads(custom_filter.search)}
|
||||||
self._original_params = {**(params or {})}
|
self._original_params = {**(params or {})}
|
||||||
|
# json.loads casts rating as an integer, expecting string
|
||||||
|
if isinstance(self._params.get('rating', None), int):
|
||||||
|
self._params['rating'] = str(self._params['rating'])
|
||||||
else:
|
else:
|
||||||
self._params = {**(params or {})}
|
self._params = {**(params or {})}
|
||||||
else:
|
else:
|
||||||
@@ -45,7 +47,8 @@ class RecipeSearch():
|
|||||||
cache.set(CACHE_KEY, self._search_prefs, timeout=10)
|
cache.set(CACHE_KEY, self._search_prefs, timeout=10)
|
||||||
else:
|
else:
|
||||||
self._search_prefs = SearchPreference()
|
self._search_prefs = SearchPreference()
|
||||||
self._string = self._params.get('query').strip() if self._params.get('query', None) else None
|
self._string = self._params.get('query').strip(
|
||||||
|
) if self._params.get('query', None) else None
|
||||||
self._rating = self._params.get('rating', None)
|
self._rating = self._params.get('rating', None)
|
||||||
self._keywords = {
|
self._keywords = {
|
||||||
'or': self._params.get('keywords_or', None) or self._params.get('keywords', None),
|
'or': self._params.get('keywords_or', None) or self._params.get('keywords', None),
|
||||||
@@ -74,7 +77,8 @@ class RecipeSearch():
|
|||||||
self._random = str2bool(self._params.get('random', False))
|
self._random = str2bool(self._params.get('random', False))
|
||||||
self._new = str2bool(self._params.get('new', False))
|
self._new = str2bool(self._params.get('new', False))
|
||||||
self._num_recent = int(self._params.get('num_recent', 0))
|
self._num_recent = int(self._params.get('num_recent', 0))
|
||||||
self._include_children = str2bool(self._params.get('include_children', None))
|
self._include_children = str2bool(
|
||||||
|
self._params.get('include_children', None))
|
||||||
self._timescooked = self._params.get('timescooked', None)
|
self._timescooked = self._params.get('timescooked', None)
|
||||||
self._cookedon = self._params.get('cookedon', None)
|
self._cookedon = self._params.get('cookedon', None)
|
||||||
self._createdon = self._params.get('createdon', None)
|
self._createdon = self._params.get('createdon', None)
|
||||||
@@ -82,9 +86,9 @@ class RecipeSearch():
|
|||||||
self._viewedon = self._params.get('viewedon', None)
|
self._viewedon = self._params.get('viewedon', None)
|
||||||
self._makenow = self._params.get('makenow', None)
|
self._makenow = self._params.get('makenow', None)
|
||||||
# this supports hidden feature to find recipes missing X ingredients
|
# this supports hidden feature to find recipes missing X ingredients
|
||||||
if type(self._makenow) == bool and self._makenow == True:
|
if isinstance(self._makenow, bool) and self._makenow == True:
|
||||||
self._makenow = 0
|
self._makenow = 0
|
||||||
elif type(self._makenow) == str and self._makenow in ["yes", "true"]:
|
elif isinstance(self._makenow, str) and self._makenow in ["yes", "true"]:
|
||||||
self._makenow = 0
|
self._makenow = 0
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
@@ -141,7 +145,7 @@ class RecipeSearch():
|
|||||||
self.unit_filters(units=self._units)
|
self.unit_filters(units=self._units)
|
||||||
self._makenow_filter(missing=self._makenow)
|
self._makenow_filter(missing=self._makenow)
|
||||||
self.string_filters(string=self._string)
|
self.string_filters(string=self._string)
|
||||||
return self._queryset.filter(space=self._request.space).distinct().order_by(*self.orderby)
|
return self._queryset.filter(space=self._request.space).order_by(*self.orderby)
|
||||||
|
|
||||||
def _sort_includes(self, *args):
|
def _sort_includes(self, *args):
|
||||||
for x in args:
|
for x in args:
|
||||||
@@ -157,7 +161,7 @@ class RecipeSearch():
|
|||||||
else:
|
else:
|
||||||
order = []
|
order = []
|
||||||
# TODO add userpreference for default sort order and replace '-favorite'
|
# TODO add userpreference for default sort order and replace '-favorite'
|
||||||
default_order = ['-name']
|
default_order = ['name']
|
||||||
# recent and new_recipe are always first; they float a few recipes to the top
|
# recent and new_recipe are always first; they float a few recipes to the top
|
||||||
if self._num_recent:
|
if self._num_recent:
|
||||||
order += ['-recent']
|
order += ['-recent']
|
||||||
@@ -166,7 +170,6 @@ class RecipeSearch():
|
|||||||
|
|
||||||
# if a sort order is provided by user - use that order
|
# if a sort order is provided by user - use that order
|
||||||
if self._sort_order:
|
if self._sort_order:
|
||||||
|
|
||||||
if not isinstance(self._sort_order, list):
|
if not isinstance(self._sort_order, list):
|
||||||
order += [self._sort_order]
|
order += [self._sort_order]
|
||||||
else:
|
else:
|
||||||
@@ -182,8 +185,10 @@ class RecipeSearch():
|
|||||||
# otherwise sort by the remaining order_by attributes or favorite by default
|
# otherwise sort by the remaining order_by attributes or favorite by default
|
||||||
else:
|
else:
|
||||||
order += default_order
|
order += default_order
|
||||||
order[:] = [Lower('name').asc() if x == 'name' else x for x in order]
|
order[:] = [Lower('name').asc() if x ==
|
||||||
order[:] = [Lower('name').desc() if x == '-name' else x for x in order]
|
'name' else x for x in order]
|
||||||
|
order[:] = [Lower('name').desc() if x ==
|
||||||
|
'-name' else x for x in order]
|
||||||
self.orderby = order
|
self.orderby = order
|
||||||
|
|
||||||
def string_filters(self, string=None):
|
def string_filters(self, string=None):
|
||||||
@@ -200,7 +205,8 @@ class RecipeSearch():
|
|||||||
for f in self._filters:
|
for f in self._filters:
|
||||||
query_filter |= f
|
query_filter |= f
|
||||||
|
|
||||||
self._queryset = self._queryset.filter(query_filter).distinct() # this creates duplicate records which can screw up other aggregates, see makenow for workaround
|
# this creates duplicate records which can screw up other aggregates, see makenow for workaround
|
||||||
|
self._queryset = self._queryset.filter(query_filter).distinct()
|
||||||
if self._fulltext_include:
|
if self._fulltext_include:
|
||||||
if self._fuzzy_match is None:
|
if self._fuzzy_match is None:
|
||||||
self._queryset = self._queryset.annotate(score=Coalesce(Max(self.search_rank), 0.0))
|
self._queryset = self._queryset.annotate(score=Coalesce(Max(self.search_rank), 0.0))
|
||||||
@@ -228,12 +234,13 @@ class RecipeSearch():
|
|||||||
default = timezone.now() - timedelta(days=100000)
|
default = timezone.now() - timedelta(days=100000)
|
||||||
else:
|
else:
|
||||||
default = timezone.now()
|
default = timezone.now()
|
||||||
self._queryset = self._queryset.annotate(lastcooked=Coalesce(
|
self._queryset = self._queryset.annotate(
|
||||||
Max(Case(When(cooklog__created_by=self._request.user, cooklog__space=self._request.space, then='cooklog__created_at'))), Value(default)))
|
lastcooked=Coalesce(Max(Case(When(cooklog__created_by=self._request.user, cooklog__space=self._request.space, then='cooklog__created_at'))), Value(default))
|
||||||
|
)
|
||||||
if cooked_date is None:
|
if cooked_date is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
cooked_date = date(*[int(x) for x in cooked_date.split('-') if x != ''])
|
cooked_date = date(*[int(x)for x in cooked_date.split('-') if x != ''])
|
||||||
|
|
||||||
if lessthan:
|
if lessthan:
|
||||||
self._queryset = self._queryset.filter(lastcooked__date__lte=cooked_date).exclude(lastcooked=default)
|
self._queryset = self._queryset.filter(lastcooked__date__lte=cooked_date).exclude(lastcooked=default)
|
||||||
@@ -254,7 +261,7 @@ class RecipeSearch():
|
|||||||
if updated_date is None:
|
if updated_date is None:
|
||||||
return
|
return
|
||||||
lessthan = '-' in updated_date[:1]
|
lessthan = '-' in updated_date[:1]
|
||||||
updated_date = date(*[int(x) for x in updated_date.split('-') if x != ''])
|
updated_date = date(*[int(x)for x in updated_date.split('-') if x != ''])
|
||||||
if lessthan:
|
if lessthan:
|
||||||
self._queryset = self._queryset.filter(updated_at__date__lte=updated_date)
|
self._queryset = self._queryset.filter(updated_at__date__lte=updated_date)
|
||||||
else:
|
else:
|
||||||
@@ -263,12 +270,13 @@ class RecipeSearch():
|
|||||||
def _viewed_on_filter(self, viewed_date=None):
|
def _viewed_on_filter(self, viewed_date=None):
|
||||||
if self._sort_includes('lastviewed') or viewed_date:
|
if self._sort_includes('lastviewed') or viewed_date:
|
||||||
longTimeAgo = timezone.now() - timedelta(days=100000)
|
longTimeAgo = timezone.now() - timedelta(days=100000)
|
||||||
self._queryset = self._queryset.annotate(lastviewed=Coalesce(
|
self._queryset = self._queryset.annotate(
|
||||||
Max(Case(When(viewlog__created_by=self._request.user, viewlog__space=self._request.space, then='viewlog__created_at'))), Value(longTimeAgo)))
|
lastviewed=Coalesce(Max(Case(When(viewlog__created_by=self._request.user, viewlog__space=self._request.space, then='viewlog__created_at'))), Value(longTimeAgo))
|
||||||
|
)
|
||||||
if viewed_date is None:
|
if viewed_date is None:
|
||||||
return
|
return
|
||||||
lessthan = '-' in viewed_date[:1]
|
lessthan = '-' in viewed_date[:1]
|
||||||
viewed_date = date(*[int(x) for x in viewed_date.split('-') if x != ''])
|
viewed_date = date(*[int(x)for x in viewed_date.split('-') if x != ''])
|
||||||
|
|
||||||
if lessthan:
|
if lessthan:
|
||||||
self._queryset = self._queryset.filter(lastviewed__date__lte=viewed_date).exclude(lastviewed=longTimeAgo)
|
self._queryset = self._queryset.filter(lastviewed__date__lte=viewed_date).exclude(lastviewed=longTimeAgo)
|
||||||
@@ -279,9 +287,11 @@ class RecipeSearch():
|
|||||||
# TODO make new days a user-setting
|
# TODO make new days a user-setting
|
||||||
if not self._new:
|
if not self._new:
|
||||||
return
|
return
|
||||||
self._queryset = (
|
self._queryset = self._queryset.annotate(
|
||||||
self._queryset.annotate(new_recipe=Case(
|
new_recipe=Case(
|
||||||
When(created_at__gte=(timezone.now() - timedelta(days=new_days)), then=('pk')), default=Value(0), ))
|
When(created_at__gte=(timezone.now() - timedelta(days=new_days)), then=('pk')),
|
||||||
|
default=Value(0),
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
def _recently_viewed(self, num_recent=None):
|
def _recently_viewed(self, num_recent=None):
|
||||||
@@ -291,19 +301,25 @@ class RecipeSearch():
|
|||||||
Max(Case(When(viewlog__created_by=self._request.user, viewlog__space=self._request.space, then='viewlog__pk'))), Value(0)))
|
Max(Case(When(viewlog__created_by=self._request.user, viewlog__space=self._request.space, then='viewlog__pk'))), Value(0)))
|
||||||
return
|
return
|
||||||
|
|
||||||
num_recent_recipes = ViewLog.objects.filter(created_by=self._request.user, space=self._request.space).values(
|
num_recent_recipes = (
|
||||||
'recipe').annotate(recent=Max('created_at')).order_by('-recent')[:num_recent]
|
ViewLog.objects.filter(created_by=self._request.user, space=self._request.space)
|
||||||
|
.values('recipe').annotate(recent=Max('created_at')).order_by('-recent')[:num_recent]
|
||||||
|
)
|
||||||
self._queryset = self._queryset.annotate(recent=Coalesce(Max(Case(When(pk__in=num_recent_recipes.values('recipe'), then='viewlog__pk'))), Value(0)))
|
self._queryset = self._queryset.annotate(recent=Coalesce(Max(Case(When(pk__in=num_recent_recipes.values('recipe'), then='viewlog__pk'))), Value(0)))
|
||||||
|
|
||||||
def _favorite_recipes(self, times_cooked=None):
|
def _favorite_recipes(self, times_cooked=None):
|
||||||
if self._sort_includes('favorite') or times_cooked:
|
if self._sort_includes('favorite') or times_cooked:
|
||||||
less_than = '-' in (times_cooked or []) or not self._sort_includes('-favorite')
|
less_than = '-' in (times_cooked or []) and not self._sort_includes('-favorite')
|
||||||
if less_than:
|
if less_than:
|
||||||
default = 1000
|
default = 1000
|
||||||
else:
|
else:
|
||||||
default = 0
|
default = 0
|
||||||
favorite_recipes = CookLog.objects.filter(created_by=self._request.user, space=self._request.space, recipe=OuterRef('pk')
|
favorite_recipes = (
|
||||||
).values('recipe').annotate(count=Count('pk', distinct=True)).values('count')
|
CookLog.objects.filter(created_by=self._request.user, space=self._request.space, recipe=OuterRef('pk'))
|
||||||
|
.values('recipe')
|
||||||
|
.annotate(count=Count('pk', distinct=True))
|
||||||
|
.values('count')
|
||||||
|
)
|
||||||
self._queryset = self._queryset.annotate(favorite=Coalesce(Subquery(favorite_recipes), default))
|
self._queryset = self._queryset.annotate(favorite=Coalesce(Subquery(favorite_recipes), default))
|
||||||
if times_cooked is None:
|
if times_cooked is None:
|
||||||
return
|
return
|
||||||
@@ -311,7 +327,7 @@ class RecipeSearch():
|
|||||||
if times_cooked == '0':
|
if times_cooked == '0':
|
||||||
self._queryset = self._queryset.filter(favorite=0)
|
self._queryset = self._queryset.filter(favorite=0)
|
||||||
elif less_than:
|
elif less_than:
|
||||||
self._queryset = self._queryset.filter(favorite__lte=int(times_cooked[1:])).exclude(favorite=0)
|
self._queryset = self._queryset.filter(favorite__lte=int(times_cooked.replace('-', ''))).exclude(favorite=0)
|
||||||
else:
|
else:
|
||||||
self._queryset = self._queryset.filter(favorite__gte=int(times_cooked))
|
self._queryset = self._queryset.filter(favorite__gte=int(times_cooked))
|
||||||
|
|
||||||
@@ -393,8 +409,9 @@ class RecipeSearch():
|
|||||||
|
|
||||||
def rating_filter(self, rating=None):
|
def rating_filter(self, rating=None):
|
||||||
if rating or self._sort_includes('rating'):
|
if rating or self._sort_includes('rating'):
|
||||||
lessthan = self._sort_includes('-rating') or '-' in (rating or [])
|
lessthan = '-' in (rating or [])
|
||||||
if lessthan:
|
reverse = 'rating' in (self._sort_order or []) and '-rating' not in (self._sort_order or [])
|
||||||
|
if lessthan or reverse:
|
||||||
default = 100
|
default = 100
|
||||||
else:
|
else:
|
||||||
default = 0
|
default = 0
|
||||||
@@ -446,7 +463,7 @@ class RecipeSearch():
|
|||||||
if not steps:
|
if not steps:
|
||||||
return
|
return
|
||||||
if not isinstance(steps, list):
|
if not isinstance(steps, list):
|
||||||
steps = [unistepsts]
|
steps = [steps]
|
||||||
self._queryset = self._queryset.filter(steps__id__in=steps)
|
self._queryset = self._queryset.filter(steps__id__in=steps)
|
||||||
|
|
||||||
def build_fulltext_filters(self, string=None):
|
def build_fulltext_filters(self, string=None):
|
||||||
@@ -503,26 +520,33 @@ class RecipeSearch():
|
|||||||
trigram += TrigramSimilarity(f, self._string)
|
trigram += TrigramSimilarity(f, self._string)
|
||||||
else:
|
else:
|
||||||
trigram = TrigramSimilarity(f, self._string)
|
trigram = TrigramSimilarity(f, self._string)
|
||||||
self._fuzzy_match = Recipe.objects.annotate(trigram=trigram).distinct(
|
self._fuzzy_match = (
|
||||||
).annotate(simularity=Max('trigram')).values('id', 'simularity').filter(simularity__gt=self._search_prefs.trigram_threshold)
|
Recipe.objects.annotate(trigram=trigram)
|
||||||
|
.distinct()
|
||||||
|
.annotate(simularity=Max('trigram'))
|
||||||
|
.values('id', 'simularity')
|
||||||
|
.filter(simularity__gt=self._search_prefs.trigram_threshold)
|
||||||
|
)
|
||||||
self._filters += [Q(pk__in=self._fuzzy_match.values('pk'))]
|
self._filters += [Q(pk__in=self._fuzzy_match.values('pk'))]
|
||||||
|
|
||||||
def _makenow_filter(self, missing=None):
|
def _makenow_filter(self, missing=None):
|
||||||
if missing is None or (type(missing) == bool and missing == False):
|
if missing is None or (isinstance(missing, bool) and missing == False):
|
||||||
return
|
return
|
||||||
shopping_users = [*self._request.user.get_shopping_share(), self._request.user]
|
shopping_users = [*self._request.user.get_shopping_share(), self._request.user]
|
||||||
|
|
||||||
onhand_filter = (
|
onhand_filter = (
|
||||||
Q(steps__ingredients__food__onhand_users__in=shopping_users) # food onhand
|
Q(steps__ingredients__food__onhand_users__in=shopping_users) # food onhand
|
||||||
| Q(steps__ingredients__food__substitute__onhand_users__in=shopping_users) # or substitute food onhand
|
# or substitute food onhand
|
||||||
| Q(steps__ingredients__food__in=self.__children_substitute_filter(shopping_users))
|
| Q(steps__ingredients__food__substitute__onhand_users__in=shopping_users)
|
||||||
| Q(steps__ingredients__food__in=self.__sibling_substitute_filter(shopping_users))
|
| Q(steps__ingredients__food__in=self.__children_substitute_filter(shopping_users))
|
||||||
|
| Q(steps__ingredients__food__in=self.__sibling_substitute_filter(shopping_users))
|
||||||
)
|
)
|
||||||
makenow_recipes = Recipe.objects.annotate(
|
makenow_recipes = Recipe.objects.annotate(
|
||||||
count_food=Count('steps__ingredients__food__pk', filter=Q(steps__ingredients__food__isnull=False), distinct=True),
|
count_food=Count('steps__ingredients__food__pk', filter=Q(steps__ingredients__food__isnull=False), distinct=True),
|
||||||
count_onhand=Count('steps__ingredients__food__pk', filter=onhand_filter, distinct=True),
|
count_onhand=Count('steps__ingredients__food__pk', filter=onhand_filter, distinct=True),
|
||||||
count_ignore_shopping=Count('steps__ingredients__food__pk', filter=Q(steps__ingredients__food__ignore_shopping=True,
|
count_ignore_shopping=Count(
|
||||||
steps__ingredients__food__recipe__isnull=True), distinct=True),
|
'steps__ingredients__food__pk', filter=Q(steps__ingredients__food__ignore_shopping=True, steps__ingredients__food__recipe__isnull=True), distinct=True
|
||||||
|
),
|
||||||
has_child_sub=Case(When(steps__ingredients__food__in=self.__children_substitute_filter(shopping_users), then=Value(1)), default=Value(0)),
|
has_child_sub=Case(When(steps__ingredients__food__in=self.__children_substitute_filter(shopping_users), then=Value(1)), default=Value(0)),
|
||||||
has_sibling_sub=Case(When(steps__ingredients__food__in=self.__sibling_substitute_filter(shopping_users), then=Value(1)), default=Value(0))
|
has_sibling_sub=Case(When(steps__ingredients__food__in=self.__sibling_substitute_filter(shopping_users), then=Value(1)), default=Value(0))
|
||||||
).annotate(missingfood=F('count_food') - F('count_onhand') - F('count_ignore_shopping')).filter(missingfood__lte=missing)
|
).annotate(missingfood=F('count_food') - F('count_onhand') - F('count_ignore_shopping')).filter(missingfood__lte=missing)
|
||||||
@@ -530,236 +554,28 @@ class RecipeSearch():
|
|||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def __children_substitute_filter(shopping_users=None):
|
def __children_substitute_filter(shopping_users=None):
|
||||||
children_onhand_subquery = Food.objects.filter(
|
children_onhand_subquery = Food.objects.filter(path__startswith=OuterRef('path'), depth__gt=OuterRef('depth'), onhand_users__in=shopping_users)
|
||||||
path__startswith=OuterRef('path'),
|
return (
|
||||||
depth__gt=OuterRef('depth'),
|
Food.objects.exclude( # list of foods that are onhand and children of: foods that are not onhand and are set to use children as substitutes
|
||||||
onhand_users__in=shopping_users
|
Q(onhand_users__in=shopping_users) | Q(ignore_shopping=True, recipe__isnull=True) | Q(substitute__onhand_users__in=shopping_users)
|
||||||
|
)
|
||||||
|
.exclude(depth=1, numchild=0)
|
||||||
|
.filter(substitute_children=True)
|
||||||
|
.annotate(child_onhand_count=Exists(children_onhand_subquery))
|
||||||
|
.filter(child_onhand_count=True)
|
||||||
)
|
)
|
||||||
return Food.objects.exclude( # list of foods that are onhand and children of: foods that are not onhand and are set to use children as substitutes
|
|
||||||
Q(onhand_users__in=shopping_users)
|
|
||||||
| Q(ignore_shopping=True, recipe__isnull=True)
|
|
||||||
| Q(substitute__onhand_users__in=shopping_users)
|
|
||||||
).exclude(depth=1, numchild=0
|
|
||||||
).filter(substitute_children=True
|
|
||||||
).annotate(child_onhand_count=Exists(children_onhand_subquery)
|
|
||||||
).filter(child_onhand_count=True)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def __sibling_substitute_filter(shopping_users=None):
|
def __sibling_substitute_filter(shopping_users=None):
|
||||||
sibling_onhand_subquery = Food.objects.filter(
|
sibling_onhand_subquery = Food.objects.filter(
|
||||||
path__startswith=Substr(OuterRef('path'), 1, Food.steplen * (OuterRef('depth') - 1)),
|
path__startswith=Substr(OuterRef('path'), 1, Food.steplen * (OuterRef('depth') - 1)), depth=OuterRef('depth'), onhand_users__in=shopping_users
|
||||||
depth=OuterRef('depth'),
|
|
||||||
onhand_users__in=shopping_users
|
|
||||||
)
|
)
|
||||||
return Food.objects.exclude( # list of foods that are onhand and siblings of: foods that are not onhand and are set to use siblings as substitutes
|
return (
|
||||||
Q(onhand_users__in=shopping_users)
|
Food.objects.exclude( # list of foods that are onhand and siblings of: foods that are not onhand and are set to use siblings as substitutes
|
||||||
| Q(ignore_shopping=True, recipe__isnull=True)
|
Q(onhand_users__in=shopping_users) | Q(ignore_shopping=True, recipe__isnull=True) | Q(substitute__onhand_users__in=shopping_users)
|
||||||
| Q(substitute__onhand_users__in=shopping_users)
|
)
|
||||||
).exclude(depth=1, numchild=0
|
.exclude(depth=1, numchild=0)
|
||||||
).filter(substitute_siblings=True
|
.filter(substitute_siblings=True)
|
||||||
).annotate(sibling_onhand=Exists(sibling_onhand_subquery)
|
.annotate(sibling_onhand=Exists(sibling_onhand_subquery))
|
||||||
).filter(sibling_onhand=True)
|
.filter(sibling_onhand=True)
|
||||||
|
|
||||||
|
|
||||||
class RecipeFacet():
|
|
||||||
class CacheEmpty(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def __init__(self, request, queryset=None, hash_key=None, cache_timeout=3600):
|
|
||||||
if hash_key is None and queryset is None:
|
|
||||||
raise ValueError(_("One of queryset or hash_key must be provided"))
|
|
||||||
|
|
||||||
self._request = request
|
|
||||||
self._queryset = queryset
|
|
||||||
self.hash_key = hash_key or str(hash(self._queryset.query))
|
|
||||||
self._SEARCH_CACHE_KEY = f"recipes_filter_{self.hash_key}"
|
|
||||||
self._cache_timeout = cache_timeout
|
|
||||||
self._cache = caches['default'].get(self._SEARCH_CACHE_KEY, {})
|
|
||||||
if self._cache is None and self._queryset is None:
|
|
||||||
raise self.CacheEmpty("No queryset provided and cache empty")
|
|
||||||
|
|
||||||
self.Keywords = self._cache.get('Keywords', None)
|
|
||||||
self.Foods = self._cache.get('Foods', None)
|
|
||||||
self.Books = self._cache.get('Books', None)
|
|
||||||
self.Ratings = self._cache.get('Ratings', None)
|
|
||||||
# TODO Move Recent to recipe annotation/serializer: requrires change in RecipeSearch(), RecipeSearchView.vue and serializer
|
|
||||||
self.Recent = self._cache.get('Recent', None)
|
|
||||||
|
|
||||||
if self._queryset is not None:
|
|
||||||
self._recipe_list = list(self._queryset.values_list('id', flat=True))
|
|
||||||
self._search_params = {
|
|
||||||
'keyword_list': self._request.query_params.getlist('keywords', []),
|
|
||||||
'food_list': self._request.query_params.getlist('foods', []),
|
|
||||||
'book_list': self._request.query_params.getlist('book', []),
|
|
||||||
'search_keywords_or': str2bool(self._request.query_params.get('keywords_or', True)),
|
|
||||||
'search_foods_or': str2bool(self._request.query_params.get('foods_or', True)),
|
|
||||||
'search_books_or': str2bool(self._request.query_params.get('books_or', True)),
|
|
||||||
'space': self._request.space,
|
|
||||||
}
|
|
||||||
elif self.hash_key is not None:
|
|
||||||
self._recipe_list = self._cache.get('recipe_list', [])
|
|
||||||
self._search_params = {
|
|
||||||
'keyword_list': self._cache.get('keyword_list', None),
|
|
||||||
'food_list': self._cache.get('food_list', None),
|
|
||||||
'book_list': self._cache.get('book_list', None),
|
|
||||||
'search_keywords_or': self._cache.get('search_keywords_or', None),
|
|
||||||
'search_foods_or': self._cache.get('search_foods_or', None),
|
|
||||||
'search_books_or': self._cache.get('search_books_or', None),
|
|
||||||
'space': self._cache.get('space', None),
|
|
||||||
}
|
|
||||||
|
|
||||||
self._cache = {
|
|
||||||
**self._search_params,
|
|
||||||
'recipe_list': self._recipe_list,
|
|
||||||
'Ratings': self.Ratings,
|
|
||||||
'Recent': self.Recent,
|
|
||||||
'Keywords': self.Keywords,
|
|
||||||
'Foods': self.Foods,
|
|
||||||
'Books': self.Books
|
|
||||||
|
|
||||||
}
|
|
||||||
caches['default'].set(self._SEARCH_CACHE_KEY, self._cache, self._cache_timeout)
|
|
||||||
|
|
||||||
def get_facets(self, from_cache=False):
|
|
||||||
if from_cache:
|
|
||||||
return {
|
|
||||||
'cache_key': self.hash_key or '',
|
|
||||||
'Ratings': self.Ratings or {},
|
|
||||||
'Recent': self.Recent or [],
|
|
||||||
'Keywords': self.Keywords or [],
|
|
||||||
'Foods': self.Foods or [],
|
|
||||||
'Books': self.Books or []
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
'cache_key': self.hash_key,
|
|
||||||
'Ratings': self.get_ratings(),
|
|
||||||
'Recent': self.get_recent(),
|
|
||||||
'Keywords': self.get_keywords(),
|
|
||||||
'Foods': self.get_foods(),
|
|
||||||
'Books': self.get_books()
|
|
||||||
}
|
|
||||||
|
|
||||||
def set_cache(self, key, value):
|
|
||||||
self._cache = {**self._cache, key: value}
|
|
||||||
caches['default'].set(
|
|
||||||
self._SEARCH_CACHE_KEY,
|
|
||||||
self._cache,
|
|
||||||
self._cache_timeout
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def get_books(self):
|
|
||||||
if self.Books is None:
|
|
||||||
self.Books = []
|
|
||||||
return self.Books
|
|
||||||
|
|
||||||
def get_keywords(self):
|
|
||||||
if self.Keywords is None:
|
|
||||||
if self._search_params['search_keywords_or']:
|
|
||||||
keywords = Keyword.objects.filter(space=self._request.space).distinct()
|
|
||||||
else:
|
|
||||||
keywords = Keyword.objects.filter(Q(recipe__in=self._recipe_list) | Q(depth=1)).filter(space=self._request.space).distinct()
|
|
||||||
|
|
||||||
# set keywords to root objects only
|
|
||||||
keywords = self._keyword_queryset(keywords)
|
|
||||||
self.Keywords = [{**x, 'children': None} if x['numchild'] > 0 else x for x in list(keywords)]
|
|
||||||
self.set_cache('Keywords', self.Keywords)
|
|
||||||
return self.Keywords
|
|
||||||
|
|
||||||
def get_foods(self):
|
|
||||||
if self.Foods is None:
|
|
||||||
# # if using an OR search, will annotate all keywords, otherwise, just those that appear in results
|
|
||||||
if self._search_params['search_foods_or']:
|
|
||||||
foods = Food.objects.filter(space=self._request.space).distinct()
|
|
||||||
else:
|
|
||||||
foods = Food.objects.filter(Q(ingredient__step__recipe__in=self._recipe_list) | Q(depth=1)).filter(space=self._request.space).distinct()
|
|
||||||
|
|
||||||
# set keywords to root objects only
|
|
||||||
foods = self._food_queryset(foods)
|
|
||||||
|
|
||||||
self.Foods = [{**x, 'children': None} if x['numchild'] > 0 else x for x in list(foods)]
|
|
||||||
self.set_cache('Foods', self.Foods)
|
|
||||||
return self.Foods
|
|
||||||
|
|
||||||
def get_books(self):
|
|
||||||
if self.Books is None:
|
|
||||||
self.Books = []
|
|
||||||
return self.Books
|
|
||||||
|
|
||||||
def get_ratings(self):
|
|
||||||
if self.Ratings is None:
|
|
||||||
if not self._request.space.demo and self._request.space.show_facet_count:
|
|
||||||
if self._queryset is None:
|
|
||||||
self._queryset = Recipe.objects.filter(id__in=self._recipe_list)
|
|
||||||
rating_qs = self._queryset.annotate(rating=Round(Avg(Case(When(cooklog__created_by=self._request.user, then='cooklog__rating'), default=Value(0)))))
|
|
||||||
self.Ratings = dict(Counter(r.rating for r in rating_qs))
|
|
||||||
else:
|
|
||||||
self.Rating = {}
|
|
||||||
self.set_cache('Ratings', self.Ratings)
|
|
||||||
return self.Ratings
|
|
||||||
|
|
||||||
def get_recent(self):
|
|
||||||
if self.Recent is None:
|
|
||||||
# TODO make days of recent recipe a setting
|
|
||||||
recent_recipes = ViewLog.objects.filter(created_by=self._request.user, space=self._request.space, created_at__gte=timezone.now() - timedelta(days=14)
|
|
||||||
).values_list('recipe__pk', flat=True)
|
|
||||||
self.Recent = list(recent_recipes)
|
|
||||||
self.set_cache('Recent', self.Recent)
|
|
||||||
return self.Recent
|
|
||||||
|
|
||||||
def add_food_children(self, id):
|
|
||||||
try:
|
|
||||||
food = Food.objects.get(id=id)
|
|
||||||
nodes = food.get_ancestors()
|
|
||||||
except Food.DoesNotExist:
|
|
||||||
return self.get_facets()
|
|
||||||
foods = self._food_queryset(food.get_children(), food)
|
|
||||||
deep_search = self.Foods
|
|
||||||
for node in nodes:
|
|
||||||
index = next((i for i, x in enumerate(deep_search) if x["id"] == node.id), None)
|
|
||||||
deep_search = deep_search[index]['children']
|
|
||||||
index = next((i for i, x in enumerate(deep_search) if x["id"] == food.id), None)
|
|
||||||
deep_search[index]['children'] = [{**x, 'children': None} if x['numchild'] > 0 else x for x in list(foods)]
|
|
||||||
self.set_cache('Foods', self.Foods)
|
|
||||||
return self.get_facets()
|
|
||||||
|
|
||||||
def add_keyword_children(self, id):
|
|
||||||
try:
|
|
||||||
keyword = Keyword.objects.get(id=id)
|
|
||||||
nodes = keyword.get_ancestors()
|
|
||||||
except Keyword.DoesNotExist:
|
|
||||||
return self.get_facets()
|
|
||||||
keywords = self._keyword_queryset(keyword.get_children(), keyword)
|
|
||||||
deep_search = self.Keywords
|
|
||||||
for node in nodes:
|
|
||||||
index = next((i for i, x in enumerate(deep_search) if x["id"] == node.id), None)
|
|
||||||
deep_search = deep_search[index]['children']
|
|
||||||
index = next((i for i, x in enumerate(deep_search) if x["id"] == keyword.id), None)
|
|
||||||
deep_search[index]['children'] = [{**x, 'children': None} if x['numchild'] > 0 else x for x in list(keywords)]
|
|
||||||
self.set_cache('Keywords', self.Keywords)
|
|
||||||
return self.get_facets()
|
|
||||||
|
|
||||||
def _recipe_count_queryset(self, field, depth=1, steplen=4):
|
|
||||||
return Recipe.objects.filter(**{f'{field}__path__startswith': OuterRef('path'), f'{field}__depth__gte': depth}, id__in=self._recipe_list, space=self._request.space
|
|
||||||
).annotate(count=Coalesce(Func('pk', function='Count'), 0)).values('count')
|
|
||||||
|
|
||||||
def _keyword_queryset(self, queryset, keyword=None):
|
|
||||||
depth = getattr(keyword, 'depth', 0) + 1
|
|
||||||
steplen = depth * Keyword.steplen
|
|
||||||
|
|
||||||
if not self._request.space.demo and self._request.space.show_facet_count:
|
|
||||||
return queryset.annotate(count=Coalesce(Subquery(self._recipe_count_queryset('keywords', depth, steplen)), 0)
|
|
||||||
).filter(depth=depth, count__gt=0
|
|
||||||
).values('id', 'name', 'count', 'numchild').order_by(Lower('name').asc())[:200]
|
|
||||||
else:
|
|
||||||
return queryset.filter(depth=depth).values('id', 'name', 'numchild').order_by(Lower('name').asc())
|
|
||||||
|
|
||||||
def _food_queryset(self, queryset, food=None):
|
|
||||||
depth = getattr(food, 'depth', 0) + 1
|
|
||||||
steplen = depth * Food.steplen
|
|
||||||
|
|
||||||
if not self._request.space.demo and self._request.space.show_facet_count:
|
|
||||||
return queryset.annotate(count=Coalesce(Subquery(self._recipe_count_queryset('steps__ingredients__food', depth, steplen)), 0)
|
|
||||||
).filter(depth__lte=depth, count__gt=0
|
|
||||||
).values('id', 'name', 'count', 'numchild').order_by(Lower('name').asc())[:200]
|
|
||||||
else:
|
|
||||||
return queryset.filter(depth__lte=depth).values('id', 'name', 'numchild').order_by(Lower('name').asc())
|
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import random
|
|
||||||
import re
|
import re
|
||||||
|
import traceback
|
||||||
from html import unescape
|
from html import unescape
|
||||||
from unicodedata import decomposition
|
|
||||||
|
|
||||||
from django.utils.dateparse import parse_duration
|
from django.utils.dateparse import parse_duration
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
@@ -10,17 +9,37 @@ from isodate.isoerror import ISO8601Error
|
|||||||
from pytube import YouTube
|
from pytube import YouTube
|
||||||
from recipe_scrapers._utils import get_host_name, get_minutes
|
from recipe_scrapers._utils import get_host_name, get_minutes
|
||||||
|
|
||||||
from cookbook.helper import recipe_url_import as helper
|
from cookbook.helper.automation_helper import AutomationEngine
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.models import Keyword, Automation
|
from cookbook.models import Automation, Keyword, PropertyType
|
||||||
|
|
||||||
|
|
||||||
# from recipe_scrapers._utils import get_minutes ## temporary until/unless upstream incorporates get_minutes() PR
|
|
||||||
|
|
||||||
|
|
||||||
def get_from_scraper(scrape, request):
|
def get_from_scraper(scrape, request):
|
||||||
# converting the scrape_me object to the existing json format based on ld+json
|
# converting the scrape_me object to the existing json format based on ld+json
|
||||||
recipe_json = {}
|
|
||||||
|
recipe_json = {
|
||||||
|
'steps': [],
|
||||||
|
'internal': True
|
||||||
|
}
|
||||||
|
keywords = []
|
||||||
|
|
||||||
|
# assign source URL
|
||||||
|
try:
|
||||||
|
source_url = scrape.canonical_url()
|
||||||
|
except Exception:
|
||||||
|
try:
|
||||||
|
source_url = scrape.url
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
if source_url:
|
||||||
|
recipe_json['source_url'] = source_url
|
||||||
|
try:
|
||||||
|
keywords.append(source_url.replace('http://', '').replace('https://', '').split('/')[0])
|
||||||
|
except Exception:
|
||||||
|
recipe_json['source_url'] = ''
|
||||||
|
|
||||||
|
automation_engine = AutomationEngine(request, source=recipe_json.get('source_url'))
|
||||||
|
# assign recipe name
|
||||||
try:
|
try:
|
||||||
recipe_json['name'] = parse_name(scrape.title()[:128] or None)
|
recipe_json['name'] = parse_name(scrape.title()[:128] or None)
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -31,6 +50,13 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
recipe_json['name'] = ''
|
recipe_json['name'] = ''
|
||||||
|
|
||||||
|
if isinstance(recipe_json['name'], list) and len(recipe_json['name']) > 0:
|
||||||
|
recipe_json['name'] = recipe_json['name'][0]
|
||||||
|
|
||||||
|
recipe_json['name'] = automation_engine.apply_regex_replace_automation(recipe_json['name'], Automation.NAME_REPLACE)
|
||||||
|
|
||||||
|
# assign recipe description
|
||||||
|
# TODO notify user about limit if reached - >256 description will be truncated
|
||||||
try:
|
try:
|
||||||
description = scrape.description() or None
|
description = scrape.description() or None
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -41,16 +67,20 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
description = ''
|
description = ''
|
||||||
|
|
||||||
recipe_json['internal'] = True
|
recipe_json['description'] = parse_description(description)
|
||||||
|
recipe_json['description'] = automation_engine.apply_regex_replace_automation(recipe_json['description'], Automation.DESCRIPTION_REPLACE)
|
||||||
|
|
||||||
|
# assign servings attributes
|
||||||
try:
|
try:
|
||||||
servings = scrape.schema.data.get('recipeYield') or 1 # dont use scrape.yields() as this will always return "x servings" or "x items", should be improved in scrapers directly
|
# dont use scrape.yields() as this will always return "x servings" or "x items", should be improved in scrapers directly
|
||||||
|
servings = scrape.schema.data.get('recipeYield') or 1
|
||||||
except Exception:
|
except Exception:
|
||||||
servings = 1
|
servings = 1
|
||||||
|
|
||||||
recipe_json['servings'] = parse_servings(servings)
|
recipe_json['servings'] = parse_servings(servings)
|
||||||
recipe_json['servings_text'] = parse_servings_text(servings)
|
recipe_json['servings_text'] = parse_servings_text(servings)
|
||||||
|
|
||||||
|
# assign time attributes
|
||||||
try:
|
try:
|
||||||
recipe_json['working_time'] = get_minutes(scrape.prep_time()) or 0
|
recipe_json['working_time'] = get_minutes(scrape.prep_time()) or 0
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -75,6 +105,7 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
# assign image
|
||||||
try:
|
try:
|
||||||
recipe_json['image'] = parse_image(scrape.image()) or None
|
recipe_json['image'] = parse_image(scrape.image()) or None
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -85,7 +116,7 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
recipe_json['image'] = ''
|
recipe_json['image'] = ''
|
||||||
|
|
||||||
keywords = []
|
# assign keywords
|
||||||
try:
|
try:
|
||||||
if scrape.schema.data.get("keywords"):
|
if scrape.schema.data.get("keywords"):
|
||||||
keywords += listify_keywords(scrape.schema.data.get("keywords"))
|
keywords += listify_keywords(scrape.schema.data.get("keywords"))
|
||||||
@@ -110,54 +141,32 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
try:
|
|
||||||
source_url = scrape.canonical_url()
|
|
||||||
except Exception:
|
|
||||||
try:
|
|
||||||
source_url = scrape.url
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
if source_url:
|
|
||||||
recipe_json['source_url'] = source_url
|
|
||||||
try:
|
|
||||||
keywords.append(source_url.replace('http://', '').replace('https://', '').split('/')[0])
|
|
||||||
except Exception:
|
|
||||||
recipe_json['source_url'] = ''
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if scrape.author():
|
if scrape.author():
|
||||||
keywords.append(scrape.author())
|
keywords.append(scrape.author())
|
||||||
except:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
try:
|
try:
|
||||||
recipe_json['keywords'] = parse_keywords(list(set(map(str.casefold, keywords))), request.space)
|
recipe_json['keywords'] = parse_keywords(list(set(map(str.casefold, keywords))), request)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
recipe_json['keywords'] = keywords
|
recipe_json['keywords'] = keywords
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(request, True)
|
ingredient_parser = IngredientParser(request, True)
|
||||||
|
|
||||||
recipe_json['steps'] = []
|
# assign steps
|
||||||
try:
|
try:
|
||||||
for i in parse_instructions(scrape.instructions()):
|
for i in parse_instructions(scrape.instructions()):
|
||||||
recipe_json['steps'].append({'instruction': i, 'ingredients': [], })
|
recipe_json['steps'].append({'instruction': i, 'ingredients': [], 'show_ingredients_table': request.user.userpreference.show_step_ingredients, })
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
if len(recipe_json['steps']) == 0:
|
if len(recipe_json['steps']) == 0:
|
||||||
recipe_json['steps'].append({'instruction': '', 'ingredients': [], })
|
recipe_json['steps'].append({'instruction': '', 'ingredients': [], })
|
||||||
|
|
||||||
parsed_description = parse_description(description)
|
if len(recipe_json['description']) > 256: # split at 256 as long descriptions don't look good on recipe cards
|
||||||
# TODO notify user about limit if reached
|
recipe_json['steps'][0]['instruction'] = f"*{recipe_json['description']}* \n\n" + recipe_json['steps'][0]['instruction']
|
||||||
# limits exist to limit the attack surface for dos style attacks
|
|
||||||
automations = Automation.objects.filter(type=Automation.DESCRIPTION_REPLACE, space=request.space, disabled=False).only('param_1', 'param_2', 'param_3').all().order_by('order')[:512]
|
|
||||||
for a in automations:
|
|
||||||
if re.match(a.param_1, (recipe_json['source_url'])[:512]):
|
|
||||||
parsed_description = re.sub(a.param_2, a.param_3, parsed_description, count=1)
|
|
||||||
|
|
||||||
if len(parsed_description) > 256: # split at 256 as long descriptions don't look good on recipe cards
|
|
||||||
recipe_json['steps'][0]['instruction'] = f'*{parsed_description}* \n\n' + recipe_json['steps'][0]['instruction']
|
|
||||||
else:
|
else:
|
||||||
recipe_json['description'] = parsed_description[:512]
|
recipe_json['description'] = recipe_json['description'][:512]
|
||||||
|
|
||||||
try:
|
try:
|
||||||
for x in scrape.ingredients():
|
for x in scrape.ingredients():
|
||||||
@@ -191,16 +200,44 @@ def get_from_scraper(scrape, request):
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
if recipe_json['source_url']:
|
try:
|
||||||
automations = Automation.objects.filter(type=Automation.INSTRUCTION_REPLACE, space=request.space, disabled=False).only('param_1', 'param_2', 'param_3').order_by('order').all()[:512]
|
recipe_json['properties'] = get_recipe_properties(request.space, scrape.schema.nutrients())
|
||||||
for a in automations:
|
print(recipe_json['properties'])
|
||||||
if re.match(a.param_1, (recipe_json['source_url'])[:512]):
|
except Exception:
|
||||||
for s in recipe_json['steps']:
|
traceback.print_exc()
|
||||||
s['instruction'] = re.sub(a.param_2, a.param_3, s['instruction'])
|
pass
|
||||||
|
|
||||||
|
for s in recipe_json['steps']:
|
||||||
|
s['instruction'] = automation_engine.apply_regex_replace_automation(s['instruction'], Automation.INSTRUCTION_REPLACE)
|
||||||
|
# re.sub(a.param_2, a.param_3, s['instruction'])
|
||||||
|
|
||||||
return recipe_json
|
return recipe_json
|
||||||
|
|
||||||
|
|
||||||
|
def get_recipe_properties(space, property_data):
|
||||||
|
# {'servingSize': '1', 'calories': '302 kcal', 'proteinContent': '7,66g', 'fatContent': '11,56g', 'carbohydrateContent': '41,33g'}
|
||||||
|
properties = {
|
||||||
|
"property-calories": "calories",
|
||||||
|
"property-carbohydrates": "carbohydrateContent",
|
||||||
|
"property-proteins": "proteinContent",
|
||||||
|
"property-fats": "fatContent",
|
||||||
|
}
|
||||||
|
recipe_properties = []
|
||||||
|
for pt in PropertyType.objects.filter(space=space, open_data_slug__in=list(properties.keys())).all():
|
||||||
|
for p in list(properties.keys()):
|
||||||
|
if pt.open_data_slug == p:
|
||||||
|
if properties[p] in property_data:
|
||||||
|
recipe_properties.append({
|
||||||
|
'property_type': {
|
||||||
|
'id': pt.id,
|
||||||
|
'name': pt.name,
|
||||||
|
},
|
||||||
|
'property_amount': parse_servings(property_data[properties[p]]) / float(property_data['servingSize']),
|
||||||
|
})
|
||||||
|
|
||||||
|
return recipe_properties
|
||||||
|
|
||||||
|
|
||||||
def get_from_youtube_scraper(url, request):
|
def get_from_youtube_scraper(url, request):
|
||||||
"""A YouTube Information Scraper."""
|
"""A YouTube Information Scraper."""
|
||||||
kw, created = Keyword.objects.get_or_create(name='YouTube', space=request.space)
|
kw, created = Keyword.objects.get_or_create(name='YouTube', space=request.space)
|
||||||
@@ -222,11 +259,14 @@ def get_from_youtube_scraper(url, request):
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# TODO add automation here
|
||||||
try:
|
try:
|
||||||
|
automation_engine = AutomationEngine(request, source=url)
|
||||||
video = YouTube(url=url)
|
video = YouTube(url=url)
|
||||||
default_recipe_json['name'] = video.title
|
default_recipe_json['name'] = automation_engine.apply_regex_replace_automation(video.title, Automation.NAME_REPLACE)
|
||||||
default_recipe_json['image'] = video.thumbnail_url
|
default_recipe_json['image'] = video.thumbnail_url
|
||||||
default_recipe_json['steps'][0]['instruction'] = video.description
|
default_recipe_json['steps'][0]['instruction'] = automation_engine.apply_regex_replace_automation(video.description, Automation.INSTRUCTION_REPLACE)
|
||||||
|
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -234,7 +274,7 @@ def get_from_youtube_scraper(url, request):
|
|||||||
|
|
||||||
|
|
||||||
def parse_name(name):
|
def parse_name(name):
|
||||||
if type(name) == list:
|
if isinstance(name, list):
|
||||||
try:
|
try:
|
||||||
name = name[0]
|
name = name[0]
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -278,16 +318,16 @@ def parse_instructions(instructions):
|
|||||||
"""
|
"""
|
||||||
instruction_list = []
|
instruction_list = []
|
||||||
|
|
||||||
if type(instructions) == list:
|
if isinstance(instructions, list):
|
||||||
for i in instructions:
|
for i in instructions:
|
||||||
if type(i) == str:
|
if isinstance(i, str):
|
||||||
instruction_list.append(clean_instruction_string(i))
|
instruction_list.append(clean_instruction_string(i))
|
||||||
else:
|
else:
|
||||||
if 'text' in i:
|
if 'text' in i:
|
||||||
instruction_list.append(clean_instruction_string(i['text']))
|
instruction_list.append(clean_instruction_string(i['text']))
|
||||||
elif 'itemListElement' in i:
|
elif 'itemListElement' in i:
|
||||||
for ile in i['itemListElement']:
|
for ile in i['itemListElement']:
|
||||||
if type(ile) == str:
|
if isinstance(ile, str):
|
||||||
instruction_list.append(clean_instruction_string(ile))
|
instruction_list.append(clean_instruction_string(ile))
|
||||||
elif 'text' in ile:
|
elif 'text' in ile:
|
||||||
instruction_list.append(clean_instruction_string(ile['text']))
|
instruction_list.append(clean_instruction_string(ile['text']))
|
||||||
@@ -303,13 +343,13 @@ def parse_image(image):
|
|||||||
# check if list of images is returned, take first if so
|
# check if list of images is returned, take first if so
|
||||||
if not image:
|
if not image:
|
||||||
return None
|
return None
|
||||||
if type(image) == list:
|
if isinstance(image, list):
|
||||||
for pic in image:
|
for pic in image:
|
||||||
if (type(pic) == str) and (pic[:4] == 'http'):
|
if (isinstance(pic, str)) and (pic[:4] == 'http'):
|
||||||
image = pic
|
image = pic
|
||||||
elif 'url' in pic:
|
elif 'url' in pic:
|
||||||
image = pic['url']
|
image = pic['url']
|
||||||
elif type(image) == dict:
|
elif isinstance(image, dict):
|
||||||
if 'url' in image:
|
if 'url' in image:
|
||||||
image = image['url']
|
image = image['url']
|
||||||
|
|
||||||
@@ -320,12 +360,12 @@ def parse_image(image):
|
|||||||
|
|
||||||
|
|
||||||
def parse_servings(servings):
|
def parse_servings(servings):
|
||||||
if type(servings) == str:
|
if isinstance(servings, str):
|
||||||
try:
|
try:
|
||||||
servings = int(re.search(r'\d+', servings).group())
|
servings = int(re.search(r'\d+', servings).group())
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
servings = 1
|
servings = 1
|
||||||
elif type(servings) == list:
|
elif isinstance(servings, list):
|
||||||
try:
|
try:
|
||||||
servings = int(re.findall(r'\b\d+\b', servings[0])[0])
|
servings = int(re.findall(r'\b\d+\b', servings[0])[0])
|
||||||
except KeyError:
|
except KeyError:
|
||||||
@@ -334,12 +374,12 @@ def parse_servings(servings):
|
|||||||
|
|
||||||
|
|
||||||
def parse_servings_text(servings):
|
def parse_servings_text(servings):
|
||||||
if type(servings) == str:
|
if isinstance(servings, str):
|
||||||
try:
|
try:
|
||||||
servings = re.sub("\d+", '', servings).strip()
|
servings = re.sub("\\d+", '', servings).strip()
|
||||||
except Exception:
|
except Exception:
|
||||||
servings = ''
|
servings = ''
|
||||||
if type(servings) == list:
|
if isinstance(servings, list):
|
||||||
try:
|
try:
|
||||||
servings = parse_servings_text(servings[1])
|
servings = parse_servings_text(servings[1])
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -356,7 +396,7 @@ def parse_time(recipe_time):
|
|||||||
recipe_time = round(iso_parse_duration(recipe_time).seconds / 60)
|
recipe_time = round(iso_parse_duration(recipe_time).seconds / 60)
|
||||||
except ISO8601Error:
|
except ISO8601Error:
|
||||||
try:
|
try:
|
||||||
if (type(recipe_time) == list and len(recipe_time) > 0):
|
if (isinstance(recipe_time, list) and len(recipe_time) > 0):
|
||||||
recipe_time = recipe_time[0]
|
recipe_time = recipe_time[0]
|
||||||
recipe_time = round(parse_duration(recipe_time).seconds / 60)
|
recipe_time = round(parse_duration(recipe_time).seconds / 60)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
@@ -365,13 +405,18 @@ def parse_time(recipe_time):
|
|||||||
return recipe_time
|
return recipe_time
|
||||||
|
|
||||||
|
|
||||||
def parse_keywords(keyword_json, space):
|
def parse_keywords(keyword_json, request):
|
||||||
keywords = []
|
keywords = []
|
||||||
|
automation_engine = AutomationEngine(request)
|
||||||
|
|
||||||
# keywords as list
|
# keywords as list
|
||||||
for kw in keyword_json:
|
for kw in keyword_json:
|
||||||
kw = normalize_string(kw)
|
kw = normalize_string(kw)
|
||||||
|
# if alias exists use that instead
|
||||||
|
|
||||||
if len(kw) != 0:
|
if len(kw) != 0:
|
||||||
if k := Keyword.objects.filter(name=kw, space=space).first():
|
automation_engine.apply_keyword_automation(kw)
|
||||||
|
if k := Keyword.objects.filter(name=kw, space=request.space).first():
|
||||||
keywords.append({'label': str(k), 'name': k.name, 'id': k.id})
|
keywords.append({'label': str(k), 'name': k.name, 'id': k.id})
|
||||||
else:
|
else:
|
||||||
keywords.append({'label': kw, 'name': kw})
|
keywords.append({'label': kw, 'name': kw})
|
||||||
@@ -382,15 +427,15 @@ def parse_keywords(keyword_json, space):
|
|||||||
def listify_keywords(keyword_list):
|
def listify_keywords(keyword_list):
|
||||||
# keywords as string
|
# keywords as string
|
||||||
try:
|
try:
|
||||||
if type(keyword_list[0]) == dict:
|
if isinstance(keyword_list[0], dict):
|
||||||
return keyword_list
|
return keyword_list
|
||||||
except (KeyError, IndexError):
|
except (KeyError, IndexError):
|
||||||
pass
|
pass
|
||||||
if type(keyword_list) == str:
|
if isinstance(keyword_list, str):
|
||||||
keyword_list = keyword_list.split(',')
|
keyword_list = keyword_list.split(',')
|
||||||
|
|
||||||
# keywords as string in list
|
# keywords as string in list
|
||||||
if (type(keyword_list) == list and len(keyword_list) == 1 and ',' in keyword_list[0]):
|
if (isinstance(keyword_list, list) and len(keyword_list) == 1 and ',' in keyword_list[0]):
|
||||||
keyword_list = keyword_list[0].split(',')
|
keyword_list = keyword_list[0].split(',')
|
||||||
return [x.strip() for x in keyword_list]
|
return [x.strip() for x in keyword_list]
|
||||||
|
|
||||||
@@ -444,13 +489,13 @@ def get_images_from_soup(soup, url):
|
|||||||
|
|
||||||
|
|
||||||
def clean_dict(input_dict, key):
|
def clean_dict(input_dict, key):
|
||||||
if type(input_dict) == dict:
|
if isinstance(input_dict, dict):
|
||||||
for x in list(input_dict):
|
for x in list(input_dict):
|
||||||
if x == key:
|
if x == key:
|
||||||
del input_dict[x]
|
del input_dict[x]
|
||||||
elif type(input_dict[x]) == dict:
|
elif isinstance(input_dict[x], dict):
|
||||||
input_dict[x] = clean_dict(input_dict[x], key)
|
input_dict[x] = clean_dict(input_dict[x], key)
|
||||||
elif type(input_dict[x]) == list:
|
elif isinstance(input_dict[x], list):
|
||||||
temp_list = []
|
temp_list = []
|
||||||
for e in input_dict[x]:
|
for e in input_dict[x]:
|
||||||
temp_list.append(clean_dict(e, key))
|
temp_list.append(clean_dict(e, key))
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django_scopes import scope, scopes_disabled
|
from django_scopes import scope, scopes_disabled
|
||||||
from oauth2_provider.contrib.rest_framework import OAuth2Authentication
|
from oauth2_provider.contrib.rest_framework import OAuth2Authentication
|
||||||
from rest_framework.authentication import TokenAuthentication
|
|
||||||
from rest_framework.authtoken.models import Token
|
|
||||||
from rest_framework.exceptions import AuthenticationFailed
|
from rest_framework.exceptions import AuthenticationFailed
|
||||||
|
|
||||||
from cookbook.views import views
|
from cookbook.views import views
|
||||||
@@ -50,7 +48,6 @@ class ScopeMiddleware:
|
|||||||
return views.no_groups(request)
|
return views.no_groups(request)
|
||||||
|
|
||||||
request.space = user_space.space
|
request.space = user_space.space
|
||||||
# with scopes_disabled():
|
|
||||||
with scope(space=request.space):
|
with scope(space=request.space):
|
||||||
return self.get_response(request)
|
return self.get_response(request)
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -1,16 +1,13 @@
|
|||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
from decimal import Decimal
|
from decimal import Decimal
|
||||||
|
|
||||||
from django.contrib.postgres.aggregates import ArrayAgg
|
|
||||||
from django.db.models import F, OuterRef, Q, Subquery, Value
|
from django.db.models import F, OuterRef, Q, Subquery, Value
|
||||||
from django.db.models.functions import Coalesce
|
from django.db.models.functions import Coalesce
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
from cookbook.helper.HelperFunctions import Round, str2bool
|
|
||||||
from cookbook.models import (Ingredient, MealPlan, Recipe, ShoppingListEntry, ShoppingListRecipe,
|
from cookbook.models import (Ingredient, MealPlan, Recipe, ShoppingListEntry, ShoppingListRecipe,
|
||||||
SupermarketCategoryRelation)
|
SupermarketCategoryRelation)
|
||||||
from recipes import settings
|
|
||||||
|
|
||||||
|
|
||||||
def shopping_helper(qs, request):
|
def shopping_helper(qs, request):
|
||||||
@@ -47,7 +44,7 @@ class RecipeShoppingEditor():
|
|||||||
self.mealplan = self._kwargs.get('mealplan', None)
|
self.mealplan = self._kwargs.get('mealplan', None)
|
||||||
if type(self.mealplan) in [int, float]:
|
if type(self.mealplan) in [int, float]:
|
||||||
self.mealplan = MealPlan.objects.filter(id=self.mealplan, space=self.space)
|
self.mealplan = MealPlan.objects.filter(id=self.mealplan, space=self.space)
|
||||||
if type(self.mealplan) == dict:
|
if isinstance(self.mealplan, dict):
|
||||||
self.mealplan = MealPlan.objects.filter(id=self.mealplan['id'], space=self.space).first()
|
self.mealplan = MealPlan.objects.filter(id=self.mealplan['id'], space=self.space).first()
|
||||||
self.id = self._kwargs.get('id', None)
|
self.id = self._kwargs.get('id', None)
|
||||||
|
|
||||||
@@ -69,11 +66,12 @@ class RecipeShoppingEditor():
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def _recipe_servings(self):
|
def _recipe_servings(self):
|
||||||
return getattr(self.recipe, 'servings', None) or getattr(getattr(self.mealplan, 'recipe', None), 'servings', None) or getattr(getattr(self._shopping_list_recipe, 'recipe', None), 'servings', None)
|
return getattr(self.recipe, 'servings', None) or getattr(getattr(self.mealplan, 'recipe', None), 'servings',
|
||||||
|
None) or getattr(getattr(self._shopping_list_recipe, 'recipe', None), 'servings', None)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _servings_factor(self):
|
def _servings_factor(self):
|
||||||
return Decimal(self.servings)/Decimal(self._recipe_servings)
|
return Decimal(self.servings) / Decimal(self._recipe_servings)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _shared_users(self):
|
def _shared_users(self):
|
||||||
@@ -90,9 +88,10 @@ class RecipeShoppingEditor():
|
|||||||
|
|
||||||
def get_recipe_ingredients(self, id, exclude_onhand=False):
|
def get_recipe_ingredients(self, id, exclude_onhand=False):
|
||||||
if exclude_onhand:
|
if exclude_onhand:
|
||||||
return Ingredient.objects.filter(step__recipe__id=id, food__ignore_shopping=False, space=self.space).exclude(food__onhand_users__id__in=[x.id for x in self._shared_users])
|
return Ingredient.objects.filter(step__recipe__id=id, food__ignore_shopping=False, space=self.space).exclude(
|
||||||
|
food__onhand_users__id__in=[x.id for x in self._shared_users])
|
||||||
else:
|
else:
|
||||||
return Ingredient.objects.filter(step__recipe__id=id, food__ignore_shopping=False, space=self.space)
|
return Ingredient.objects.filter(step__recipe__id=id, food__ignore_shopping=False, space=self.space)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _include_related(self):
|
def _include_related(self):
|
||||||
@@ -109,7 +108,7 @@ class RecipeShoppingEditor():
|
|||||||
self.servings = float(servings)
|
self.servings = float(servings)
|
||||||
|
|
||||||
if mealplan := kwargs.get('mealplan', None):
|
if mealplan := kwargs.get('mealplan', None):
|
||||||
if type(mealplan) == dict:
|
if isinstance(mealplan, dict):
|
||||||
self.mealplan = MealPlan.objects.filter(id=mealplan['id'], space=self.space).first()
|
self.mealplan = MealPlan.objects.filter(id=mealplan['id'], space=self.space).first()
|
||||||
else:
|
else:
|
||||||
self.mealplan = mealplan
|
self.mealplan = mealplan
|
||||||
@@ -170,14 +169,14 @@ class RecipeShoppingEditor():
|
|||||||
try:
|
try:
|
||||||
self._shopping_list_recipe.delete()
|
self._shopping_list_recipe.delete()
|
||||||
return True
|
return True
|
||||||
except:
|
except BaseException:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _add_ingredients(self, ingredients=None):
|
def _add_ingredients(self, ingredients=None):
|
||||||
if not ingredients:
|
if not ingredients:
|
||||||
return
|
return
|
||||||
elif type(ingredients) == list:
|
elif isinstance(ingredients, list):
|
||||||
ingredients = Ingredient.objects.filter(id__in=ingredients)
|
ingredients = Ingredient.objects.filter(id__in=ingredients, food__ignore_shopping=False)
|
||||||
existing = self._shopping_list_recipe.entries.filter(ingredient__in=ingredients).values_list('ingredient__pk', flat=True)
|
existing = self._shopping_list_recipe.entries.filter(ingredient__in=ingredients).values_list('ingredient__pk', flat=True)
|
||||||
add_ingredients = ingredients.exclude(id__in=existing)
|
add_ingredients = ingredients.exclude(id__in=existing)
|
||||||
|
|
||||||
@@ -199,120 +198,3 @@ class RecipeShoppingEditor():
|
|||||||
to_delete = self._shopping_list_recipe.entries.exclude(ingredient__in=ingredients)
|
to_delete = self._shopping_list_recipe.entries.exclude(ingredient__in=ingredients)
|
||||||
ShoppingListEntry.objects.filter(id__in=to_delete).delete()
|
ShoppingListEntry.objects.filter(id__in=to_delete).delete()
|
||||||
self._shopping_list_recipe = self.get_shopping_list_recipe(self.id, self.created_by, self.space)
|
self._shopping_list_recipe = self.get_shopping_list_recipe(self.id, self.created_by, self.space)
|
||||||
|
|
||||||
|
|
||||||
# # TODO refactor as class
|
|
||||||
# def list_from_recipe(list_recipe=None, recipe=None, mealplan=None, servings=None, ingredients=None, created_by=None, space=None, append=False):
|
|
||||||
# """
|
|
||||||
# Creates ShoppingListRecipe and associated ShoppingListEntrys from a recipe or a meal plan with a recipe
|
|
||||||
# :param list_recipe: Modify an existing ShoppingListRecipe
|
|
||||||
# :param recipe: Recipe to use as list of ingredients. One of [recipe, mealplan] are required
|
|
||||||
# :param mealplan: alternatively use a mealplan recipe as source of ingredients
|
|
||||||
# :param servings: Optional: Number of servings to use to scale shoppinglist. If servings = 0 an existing recipe list will be deleted
|
|
||||||
# :param ingredients: Ingredients, list of ingredient IDs to include on the shopping list. When not provided all ingredients will be used
|
|
||||||
# :param append: If False will remove any entries not included with ingredients, when True will append ingredients to the shopping list
|
|
||||||
# """
|
|
||||||
# r = recipe or getattr(mealplan, 'recipe', None) or getattr(list_recipe, 'recipe', None)
|
|
||||||
# if not r:
|
|
||||||
# raise ValueError(_("You must supply a recipe or mealplan"))
|
|
||||||
|
|
||||||
# created_by = created_by or getattr(ShoppingListEntry.objects.filter(list_recipe=list_recipe).first(), 'created_by', None)
|
|
||||||
# if not created_by:
|
|
||||||
# raise ValueError(_("You must supply a created_by"))
|
|
||||||
|
|
||||||
# try:
|
|
||||||
# servings = float(servings)
|
|
||||||
# except (ValueError, TypeError):
|
|
||||||
# servings = getattr(mealplan, 'servings', 1.0)
|
|
||||||
|
|
||||||
# servings_factor = servings / r.servings
|
|
||||||
|
|
||||||
# shared_users = list(created_by.get_shopping_share())
|
|
||||||
# shared_users.append(created_by)
|
|
||||||
# if list_recipe:
|
|
||||||
# created = False
|
|
||||||
# else:
|
|
||||||
# list_recipe = ShoppingListRecipe.objects.create(recipe=r, mealplan=mealplan, servings=servings)
|
|
||||||
# created = True
|
|
||||||
|
|
||||||
# related_step_ing = []
|
|
||||||
# if servings == 0 and not created:
|
|
||||||
# list_recipe.delete()
|
|
||||||
# return []
|
|
||||||
# elif ingredients:
|
|
||||||
# ingredients = Ingredient.objects.filter(pk__in=ingredients, space=space)
|
|
||||||
# else:
|
|
||||||
# ingredients = Ingredient.objects.filter(step__recipe=r, food__ignore_shopping=False, space=space)
|
|
||||||
|
|
||||||
# if exclude_onhand := created_by.userpreference.mealplan_autoexclude_onhand:
|
|
||||||
# ingredients = ingredients.exclude(food__onhand_users__id__in=[x.id for x in shared_users])
|
|
||||||
|
|
||||||
# if related := created_by.userpreference.mealplan_autoinclude_related:
|
|
||||||
# # TODO: add levels of related recipes (related recipes of related recipes) to use when auto-adding mealplans
|
|
||||||
# related_recipes = r.get_related_recipes()
|
|
||||||
|
|
||||||
# for x in related_recipes:
|
|
||||||
# # related recipe is a Step serving size is driven by recipe serving size
|
|
||||||
# # TODO once/if Steps can have a serving size this needs to be refactored
|
|
||||||
# if exclude_onhand:
|
|
||||||
# # if steps are used more than once in a recipe or subrecipe - I don' think this results in the desired behavior
|
|
||||||
# related_step_ing += Ingredient.objects.filter(step__recipe=x, space=space).exclude(food__onhand_users__id__in=[x.id for x in shared_users]).values_list('id', flat=True)
|
|
||||||
# else:
|
|
||||||
# related_step_ing += Ingredient.objects.filter(step__recipe=x, space=space).values_list('id', flat=True)
|
|
||||||
|
|
||||||
# x_ing = []
|
|
||||||
# if ingredients.filter(food__recipe=x).exists():
|
|
||||||
# for ing in ingredients.filter(food__recipe=x):
|
|
||||||
# if exclude_onhand:
|
|
||||||
# x_ing = Ingredient.objects.filter(step__recipe=x, food__ignore_shopping=False, space=space).exclude(food__onhand_users__id__in=[x.id for x in shared_users])
|
|
||||||
# else:
|
|
||||||
# x_ing = Ingredient.objects.filter(step__recipe=x, food__ignore_shopping=False, space=space).exclude(food__ignore_shopping=True)
|
|
||||||
# for i in [x for x in x_ing]:
|
|
||||||
# ShoppingListEntry.objects.create(
|
|
||||||
# list_recipe=list_recipe,
|
|
||||||
# food=i.food,
|
|
||||||
# unit=i.unit,
|
|
||||||
# ingredient=i,
|
|
||||||
# amount=i.amount * Decimal(servings_factor),
|
|
||||||
# created_by=created_by,
|
|
||||||
# space=space,
|
|
||||||
# )
|
|
||||||
# # dont' add food to the shopping list that are actually recipes that will be added as ingredients
|
|
||||||
# ingredients = ingredients.exclude(food__recipe=x)
|
|
||||||
|
|
||||||
# add_ingredients = list(ingredients.values_list('id', flat=True)) + related_step_ing
|
|
||||||
# if not append:
|
|
||||||
# existing_list = ShoppingListEntry.objects.filter(list_recipe=list_recipe)
|
|
||||||
# # delete shopping list entries not included in ingredients
|
|
||||||
# existing_list.exclude(ingredient__in=ingredients).delete()
|
|
||||||
# # add shopping list entries that did not previously exist
|
|
||||||
# add_ingredients = set(add_ingredients) - set(existing_list.values_list('ingredient__id', flat=True))
|
|
||||||
# add_ingredients = Ingredient.objects.filter(id__in=add_ingredients, space=space)
|
|
||||||
|
|
||||||
# # if servings have changed, update the ShoppingListRecipe and existing Entries
|
|
||||||
# if servings <= 0:
|
|
||||||
# servings = 1
|
|
||||||
|
|
||||||
# if not created and list_recipe.servings != servings:
|
|
||||||
# update_ingredients = set(ingredients.values_list('id', flat=True)) - set(add_ingredients.values_list('id', flat=True))
|
|
||||||
# list_recipe.servings = servings
|
|
||||||
# list_recipe.save()
|
|
||||||
# for sle in ShoppingListEntry.objects.filter(list_recipe=list_recipe, ingredient__id__in=update_ingredients):
|
|
||||||
# sle.amount = sle.ingredient.amount * Decimal(servings_factor)
|
|
||||||
# sle.save()
|
|
||||||
|
|
||||||
# # add any missing Entries
|
|
||||||
# for i in [x for x in add_ingredients if x.food]:
|
|
||||||
|
|
||||||
# ShoppingListEntry.objects.create(
|
|
||||||
# list_recipe=list_recipe,
|
|
||||||
# food=i.food,
|
|
||||||
# unit=i.unit,
|
|
||||||
# ingredient=i,
|
|
||||||
# amount=i.amount * Decimal(servings_factor),
|
|
||||||
# created_by=created_by,
|
|
||||||
# space=space,
|
|
||||||
# )
|
|
||||||
|
|
||||||
# # return all shopping list items
|
|
||||||
# return list_recipe
|
|
||||||
@@ -2,7 +2,6 @@ from gettext import gettext as _
|
|||||||
|
|
||||||
import bleach
|
import bleach
|
||||||
import markdown as md
|
import markdown as md
|
||||||
from bleach_allowlist import markdown_attrs, markdown_tags
|
|
||||||
from jinja2 import Template, TemplateSyntaxError, UndefinedError
|
from jinja2 import Template, TemplateSyntaxError, UndefinedError
|
||||||
from markdown.extensions.tables import TableExtension
|
from markdown.extensions.tables import TableExtension
|
||||||
|
|
||||||
@@ -53,9 +52,17 @@ class IngredientObject(object):
|
|||||||
def render_instructions(step): # TODO deduplicate markdown cleanup code
|
def render_instructions(step): # TODO deduplicate markdown cleanup code
|
||||||
instructions = step.instruction
|
instructions = step.instruction
|
||||||
|
|
||||||
tags = markdown_tags + [
|
tags = {
|
||||||
'pre', 'table', 'td', 'tr', 'th', 'tbody', 'style', 'thead', 'img'
|
"h1", "h2", "h3", "h4", "h5", "h6",
|
||||||
]
|
"b", "i", "strong", "em", "tt",
|
||||||
|
"p", "br",
|
||||||
|
"span", "div", "blockquote", "code", "pre", "hr",
|
||||||
|
"ul", "ol", "li", "dd", "dt",
|
||||||
|
"img",
|
||||||
|
"a",
|
||||||
|
"sub", "sup",
|
||||||
|
'pre', 'table', 'td', 'tr', 'th', 'tbody', 'style', 'thead'
|
||||||
|
}
|
||||||
parsed_md = md.markdown(
|
parsed_md = md.markdown(
|
||||||
instructions,
|
instructions,
|
||||||
extensions=[
|
extensions=[
|
||||||
@@ -63,7 +70,11 @@ def render_instructions(step): # TODO deduplicate markdown cleanup code
|
|||||||
UrlizeExtension(), MarkdownFormatExtension()
|
UrlizeExtension(), MarkdownFormatExtension()
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
markdown_attrs['*'] = markdown_attrs['*'] + ['class', 'width', 'height']
|
markdown_attrs = {
|
||||||
|
"*": ["id", "class", 'width', 'height'],
|
||||||
|
"img": ["src", "alt", "title"],
|
||||||
|
"a": ["href", "alt", "title"],
|
||||||
|
}
|
||||||
|
|
||||||
instructions = bleach.clean(parsed_md, tags, markdown_attrs)
|
instructions = bleach.clean(parsed_md, tags, markdown_attrs)
|
||||||
|
|
||||||
|
|||||||
141
cookbook/helper/unit_conversion_helper.py
Normal file
141
cookbook/helper/unit_conversion_helper.py
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
from django.core.cache import caches
|
||||||
|
from decimal import Decimal
|
||||||
|
|
||||||
|
from cookbook.helper.cache_helper import CacheHelper
|
||||||
|
from cookbook.models import Ingredient, Unit
|
||||||
|
|
||||||
|
CONVERSION_TABLE = {
|
||||||
|
'weight': {
|
||||||
|
'g': 1000,
|
||||||
|
'kg': 1,
|
||||||
|
'ounce': 35.274,
|
||||||
|
'pound': 2.20462
|
||||||
|
},
|
||||||
|
'volume': {
|
||||||
|
'ml': 1000,
|
||||||
|
'l': 1,
|
||||||
|
'fluid_ounce': 33.814,
|
||||||
|
'pint': 2.11338,
|
||||||
|
'quart': 1.05669,
|
||||||
|
'gallon': 0.264172,
|
||||||
|
'tbsp': 67.628,
|
||||||
|
'tsp': 202.884,
|
||||||
|
'imperial_fluid_ounce': 35.1951,
|
||||||
|
'imperial_pint': 1.75975,
|
||||||
|
'imperial_quart': 0.879877,
|
||||||
|
'imperial_gallon': 0.219969,
|
||||||
|
'imperial_tbsp': 56.3121,
|
||||||
|
'imperial_tsp': 168.936,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
BASE_UNITS_WEIGHT = list(CONVERSION_TABLE['weight'].keys())
|
||||||
|
BASE_UNITS_VOLUME = list(CONVERSION_TABLE['volume'].keys())
|
||||||
|
|
||||||
|
|
||||||
|
class ConversionException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class UnitConversionHelper:
|
||||||
|
space = None
|
||||||
|
|
||||||
|
def __init__(self, space):
|
||||||
|
"""
|
||||||
|
Initializes unit conversion helper
|
||||||
|
:param space: space to perform conversions on
|
||||||
|
"""
|
||||||
|
self.space = space
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def convert_from_to(from_unit, to_unit, amount):
|
||||||
|
"""
|
||||||
|
Convert from one base unit to another. Throws ConversionException if trying to convert between different systems (weight/volume) or if units are not supported.
|
||||||
|
:param from_unit: str unit to convert from
|
||||||
|
:param to_unit: str unit to convert to
|
||||||
|
:param amount: amount to convert
|
||||||
|
:return: Decimal converted amount
|
||||||
|
"""
|
||||||
|
system = None
|
||||||
|
if from_unit in BASE_UNITS_WEIGHT and to_unit in BASE_UNITS_WEIGHT:
|
||||||
|
system = 'weight'
|
||||||
|
if from_unit in BASE_UNITS_VOLUME and to_unit in BASE_UNITS_VOLUME:
|
||||||
|
system = 'volume'
|
||||||
|
|
||||||
|
if not system:
|
||||||
|
raise ConversionException('Trying to convert units not existing or not in one unit system (weight/volume)')
|
||||||
|
|
||||||
|
return Decimal(amount / Decimal(CONVERSION_TABLE[system][from_unit] / CONVERSION_TABLE[system][to_unit]))
|
||||||
|
|
||||||
|
def base_conversions(self, ingredient_list):
|
||||||
|
"""
|
||||||
|
Calculates all possible base unit conversions for each ingredient give.
|
||||||
|
Converts to all common base units IF they exist in the unit database of the space.
|
||||||
|
For useful results all ingredients passed should be of the same food, otherwise filtering afterwards might be required.
|
||||||
|
:param ingredient_list: list of ingredients to convert
|
||||||
|
:return: ingredient list with appended conversions
|
||||||
|
"""
|
||||||
|
base_conversion_ingredient_list = ingredient_list.copy()
|
||||||
|
for i in ingredient_list:
|
||||||
|
try:
|
||||||
|
conversion_unit = i.unit.name
|
||||||
|
if i.unit.base_unit:
|
||||||
|
conversion_unit = i.unit.base_unit
|
||||||
|
|
||||||
|
# TODO allow setting which units to convert to? possibly only once conversions become visible
|
||||||
|
units = caches['default'].get(CacheHelper(self.space).BASE_UNITS_CACHE_KEY, None)
|
||||||
|
if not units:
|
||||||
|
units = Unit.objects.filter(space=self.space, base_unit__in=(BASE_UNITS_VOLUME + BASE_UNITS_WEIGHT)).all()
|
||||||
|
caches['default'].set(CacheHelper(self.space).BASE_UNITS_CACHE_KEY, units, 60 * 60) # cache is cleared on unit save signal so long duration is fine
|
||||||
|
|
||||||
|
for u in units:
|
||||||
|
try:
|
||||||
|
ingredient = Ingredient(amount=self.convert_from_to(conversion_unit, u.base_unit, i.amount), unit=u, food=ingredient_list[0].food, )
|
||||||
|
if not any((x.unit.name == ingredient.unit.name or x.unit.base_unit == ingredient.unit.name) for x in base_conversion_ingredient_list):
|
||||||
|
base_conversion_ingredient_list.append(ingredient)
|
||||||
|
except ConversionException:
|
||||||
|
pass
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return base_conversion_ingredient_list
|
||||||
|
|
||||||
|
def get_conversions(self, ingredient):
|
||||||
|
"""
|
||||||
|
Converts an ingredient to all possible conversions based on the custom unit conversion database.
|
||||||
|
After that passes conversion to UnitConversionHelper.base_conversions() to get all base conversions possible.
|
||||||
|
:param ingredient: Ingredient object
|
||||||
|
:return: list of ingredients with all possible custom and base conversions
|
||||||
|
"""
|
||||||
|
conversions = [ingredient]
|
||||||
|
if ingredient.unit:
|
||||||
|
for c in ingredient.unit.unit_conversion_base_relation.all():
|
||||||
|
if c.space == self.space:
|
||||||
|
r = self._uc_convert(c, ingredient.amount, ingredient.unit, ingredient.food)
|
||||||
|
if r and r not in conversions:
|
||||||
|
conversions.append(r)
|
||||||
|
for c in ingredient.unit.unit_conversion_converted_relation.all():
|
||||||
|
if c.space == self.space:
|
||||||
|
r = self._uc_convert(c, ingredient.amount, ingredient.unit, ingredient.food)
|
||||||
|
if r and r not in conversions:
|
||||||
|
conversions.append(r)
|
||||||
|
|
||||||
|
conversions = self.base_conversions(conversions)
|
||||||
|
|
||||||
|
return conversions
|
||||||
|
|
||||||
|
def _uc_convert(self, uc, amount, unit, food):
|
||||||
|
"""
|
||||||
|
Helper to calculate values for custom unit conversions.
|
||||||
|
Converts given base values using the passed UnitConversion object into a converted Ingredient
|
||||||
|
:param uc: UnitConversion object
|
||||||
|
:param amount: base amount
|
||||||
|
:param unit: base unit
|
||||||
|
:param food: base food
|
||||||
|
:return: converted ingredient object from base amount/unit/food
|
||||||
|
"""
|
||||||
|
if uc.food is None or uc.food == food:
|
||||||
|
if unit == uc.base_unit:
|
||||||
|
return Ingredient(amount=amount * (uc.converted_amount / uc.base_amount), unit=uc.converted_unit, food=food, space=self.space)
|
||||||
|
else:
|
||||||
|
return Ingredient(amount=amount * (uc.base_amount / uc.converted_amount), unit=uc.base_unit, food=food, space=self.space)
|
||||||
@@ -36,7 +36,7 @@ class ChefTap(Integration):
|
|||||||
|
|
||||||
recipe = Recipe.objects.create(name=title, created_by=self.request.user, internal=True, space=self.request.space, )
|
recipe = Recipe.objects.create(name=title, created_by=self.request.user, internal=True, space=self.request.space, )
|
||||||
|
|
||||||
step = Step.objects.create(instruction='\n'.join(directions), space=self.request.space,)
|
step = Step.objects.create(instruction='\n'.join(directions), space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,)
|
||||||
|
|
||||||
if source_url != '':
|
if source_url != '':
|
||||||
step.instruction += '\n' + source_url
|
step.instruction += '\n' + source_url
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ class Chowdown(Integration):
|
|||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction='\n'.join(directions) + '\n\n' + '\n'.join(descriptions), space=self.request.space,
|
instruction='\n'.join(directions) + '\n\n' + '\n'.join(descriptions), space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
|
|||||||
@@ -1,20 +1,15 @@
|
|||||||
import base64
|
|
||||||
import gzip
|
|
||||||
import json
|
|
||||||
import re
|
import re
|
||||||
from gettext import gettext as _
|
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
import validators
|
import validators
|
||||||
import yaml
|
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.helper.recipe_url_import import (get_from_scraper, get_images_from_soup,
|
from cookbook.helper.recipe_url_import import (get_from_scraper, get_images_from_soup,
|
||||||
iso_duration_to_minutes)
|
iso_duration_to_minutes)
|
||||||
from cookbook.helper.scrapers.scrapers import text_scraper
|
from cookbook.helper.scrapers.scrapers import text_scraper
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Keyword, Recipe, Step
|
from cookbook.models import Ingredient, Recipe, Step
|
||||||
|
|
||||||
|
|
||||||
class CookBookApp(Integration):
|
class CookBookApp(Integration):
|
||||||
@@ -25,7 +20,6 @@ class CookBookApp(Integration):
|
|||||||
def get_recipe_from_file(self, file):
|
def get_recipe_from_file(self, file):
|
||||||
recipe_html = file.getvalue().decode("utf-8")
|
recipe_html = file.getvalue().decode("utf-8")
|
||||||
|
|
||||||
# recipe_json, recipe_tree, html_data, images = get_recipe_from_source(recipe_html, 'CookBookApp', self.request)
|
|
||||||
scrape = text_scraper(text=recipe_html)
|
scrape = text_scraper(text=recipe_html)
|
||||||
recipe_json = get_from_scraper(scrape, self.request)
|
recipe_json = get_from_scraper(scrape, self.request)
|
||||||
images = list(dict.fromkeys(get_images_from_soup(scrape.soup, None)))
|
images = list(dict.fromkeys(get_images_from_soup(scrape.soup, None)))
|
||||||
@@ -37,7 +31,7 @@ class CookBookApp(Integration):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
recipe.servings = re.findall('([0-9])+', recipe_json['recipeYield'])[0]
|
recipe.servings = re.findall('([0-9])+', recipe_json['recipeYield'])[0]
|
||||||
except Exception as e:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -47,7 +41,8 @@ class CookBookApp(Integration):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
# assuming import files only contain single step
|
# assuming import files only contain single step
|
||||||
step = Step.objects.create(instruction=recipe_json['steps'][0]['instruction'], space=self.request.space, )
|
step = Step.objects.create(instruction=recipe_json['steps'][0]['instruction'], space=self.request.space,
|
||||||
|
show_ingredients_table=self.request.user.userpreference.show_step_ingredients, )
|
||||||
|
|
||||||
if 'nutrition' in recipe_json:
|
if 'nutrition' in recipe_json:
|
||||||
step.instruction = step.instruction + '\n\n' + recipe_json['nutrition']
|
step.instruction = step.instruction + '\n\n' + recipe_json['nutrition']
|
||||||
@@ -62,7 +57,7 @@ class CookBookApp(Integration):
|
|||||||
if unit := ingredient.get('unit', None):
|
if unit := ingredient.get('unit', None):
|
||||||
u = ingredient_parser.get_unit(unit.get('name', None))
|
u = ingredient_parser.get_unit(unit.get('name', None))
|
||||||
step.ingredients.add(Ingredient.objects.create(
|
step.ingredients.add(Ingredient.objects.create(
|
||||||
food=f, unit=u, amount=ingredient.get('amount', None), note=ingredient.get('note', None), original_text=ingredient.get('original_text', None), space=self.request.space,
|
food=f, unit=u, amount=ingredient.get('amount', None), note=ingredient.get('note', None), original_text=ingredient.get('original_text', None), space=self.request.space,
|
||||||
))
|
))
|
||||||
|
|
||||||
if len(images) > 0:
|
if len(images) > 0:
|
||||||
|
|||||||
@@ -1,17 +1,12 @@
|
|||||||
import base64
|
|
||||||
import json
|
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
|
||||||
from gettext import gettext as _
|
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
import validators
|
import validators
|
||||||
from lxml import etree
|
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.helper.recipe_url_import import parse_servings, parse_time, parse_servings_text
|
from cookbook.helper.recipe_url_import import parse_servings, parse_servings_text, parse_time
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Keyword, Recipe, Step
|
from cookbook.models import Ingredient, Recipe, Step
|
||||||
|
|
||||||
|
|
||||||
class Cookmate(Integration):
|
class Cookmate(Integration):
|
||||||
@@ -50,7 +45,7 @@ class Cookmate(Integration):
|
|||||||
for step in recipe_text.getchildren():
|
for step in recipe_text.getchildren():
|
||||||
if step.text:
|
if step.text:
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction=step.text.strip(), space=self.request.space,
|
instruction=step.text.strip(), space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import re
|
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
|
|
||||||
@@ -26,12 +25,13 @@ class CopyMeThat(Integration):
|
|||||||
except AttributeError:
|
except AttributeError:
|
||||||
source = None
|
source = None
|
||||||
|
|
||||||
recipe = Recipe.objects.create(name=file.find("div", {"id": "name"}).text.strip()[:128], source_url=source, created_by=self.request.user, internal=True, space=self.request.space, )
|
recipe = Recipe.objects.create(name=file.find("div", {"id": "name"}).text.strip(
|
||||||
|
)[:128], source_url=source, created_by=self.request.user, internal=True, space=self.request.space, )
|
||||||
|
|
||||||
for category in file.find_all("span", {"class": "recipeCategory"}):
|
for category in file.find_all("span", {"class": "recipeCategory"}):
|
||||||
keyword, created = Keyword.objects.get_or_create(name=category.text, space=self.request.space)
|
keyword, created = Keyword.objects.get_or_create(name=category.text, space=self.request.space)
|
||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
recipe.servings = parse_servings(file.find("a", {"id": "recipeYield"}).text.strip())
|
recipe.servings = parse_servings(file.find("a", {"id": "recipeYield"}).text.strip())
|
||||||
recipe.working_time = iso_duration_to_minutes(file.find("span", {"meta": "prepTime"}).text.strip())
|
recipe.working_time = iso_duration_to_minutes(file.find("span", {"meta": "prepTime"}).text.strip())
|
||||||
@@ -51,7 +51,7 @@ class CopyMeThat(Integration):
|
|||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
step = Step.objects.create(instruction='', space=self.request.space, )
|
step = Step.objects.create(instruction='', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients, )
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
|
|
||||||
@@ -61,7 +61,14 @@ class CopyMeThat(Integration):
|
|||||||
if not isinstance(ingredient, Tag) or not ingredient.text.strip() or "recipeIngredient_spacer" in ingredient['class']:
|
if not isinstance(ingredient, Tag) or not ingredient.text.strip() or "recipeIngredient_spacer" in ingredient['class']:
|
||||||
continue
|
continue
|
||||||
if any(x in ingredient['class'] for x in ["recipeIngredient_subheader", "recipeIngredient_note"]):
|
if any(x in ingredient['class'] for x in ["recipeIngredient_subheader", "recipeIngredient_note"]):
|
||||||
step.ingredients.add(Ingredient.objects.create(is_header=True, note=ingredient.text.strip()[:256], original_text=ingredient.text.strip(), space=self.request.space, ))
|
step.ingredients.add(
|
||||||
|
Ingredient.objects.create(
|
||||||
|
is_header=True,
|
||||||
|
note=ingredient.text.strip()[
|
||||||
|
:256],
|
||||||
|
original_text=ingredient.text.strip(),
|
||||||
|
space=self.request.space,
|
||||||
|
))
|
||||||
else:
|
else:
|
||||||
amount, unit, food, note = ingredient_parser.parse(ingredient.text.strip())
|
amount, unit, food, note = ingredient_parser.parse(ingredient.text.strip())
|
||||||
f = ingredient_parser.get_food(food)
|
f = ingredient_parser.get_food(food)
|
||||||
@@ -78,7 +85,7 @@ class CopyMeThat(Integration):
|
|||||||
step.save()
|
step.save()
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
step = Step.objects.create(instruction='', space=self.request.space, )
|
step = Step.objects.create(instruction='', space=self.request.space, )
|
||||||
|
|
||||||
step.name = instruction.text.strip()[:128]
|
step.name = instruction.text.strip()[:128]
|
||||||
else:
|
else:
|
||||||
step.instruction += instruction.text.strip() + ' \n\n'
|
step.instruction += instruction.text.strip() + ' \n\n'
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import json
|
import json
|
||||||
|
import traceback
|
||||||
from io import BytesIO, StringIO
|
from io import BytesIO, StringIO
|
||||||
from re import match
|
from re import match
|
||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
@@ -19,7 +20,10 @@ class Default(Integration):
|
|||||||
recipe = self.decode_recipe(recipe_string)
|
recipe = self.decode_recipe(recipe_string)
|
||||||
images = list(filter(lambda v: match('image.*', v), recipe_zip.namelist()))
|
images = list(filter(lambda v: match('image.*', v), recipe_zip.namelist()))
|
||||||
if images:
|
if images:
|
||||||
self.import_recipe_image(recipe, BytesIO(recipe_zip.read(images[0])), filetype=get_filetype(images[0]))
|
try:
|
||||||
|
self.import_recipe_image(recipe, BytesIO(recipe_zip.read(images[0])), filetype=get_filetype(images[0]))
|
||||||
|
except AttributeError:
|
||||||
|
traceback.print_exc()
|
||||||
return recipe
|
return recipe
|
||||||
|
|
||||||
def decode_recipe(self, string):
|
def decode_recipe(self, string):
|
||||||
@@ -54,7 +58,7 @@ class Default(Integration):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
recipe_zip_obj.writestr(f'image{get_filetype(r.image.file.name)}', r.image.file.read())
|
recipe_zip_obj.writestr(f'image{get_filetype(r.image.file.name)}', r.image.file.read())
|
||||||
except ValueError:
|
except (ValueError, FileNotFoundError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
recipe_zip_obj.close()
|
recipe_zip_obj.close()
|
||||||
@@ -67,4 +71,4 @@ class Default(Integration):
|
|||||||
|
|
||||||
export_zip_obj.close()
|
export_zip_obj.close()
|
||||||
|
|
||||||
return [[ self.get_export_file_name(), export_zip_stream.getvalue() ]]
|
return [[self.get_export_file_name(), export_zip_stream.getvalue()]]
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ class Domestica(Integration):
|
|||||||
recipe.save()
|
recipe.save()
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction=file['directions'], space=self.request.space,
|
instruction=file['directions'], space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
if file['source'] != '':
|
if file['source'] != '':
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import traceback
|
|
||||||
import datetime
|
import datetime
|
||||||
import traceback
|
import traceback
|
||||||
import uuid
|
import uuid
|
||||||
@@ -18,8 +17,7 @@ from lxml import etree
|
|||||||
|
|
||||||
from cookbook.helper.image_processing import handle_image
|
from cookbook.helper.image_processing import handle_image
|
||||||
from cookbook.models import Keyword, Recipe
|
from cookbook.models import Keyword, Recipe
|
||||||
from recipes.settings import DEBUG
|
from recipes.settings import DEBUG, EXPORT_FILE_CACHE_DURATION
|
||||||
from recipes.settings import EXPORT_FILE_CACHE_DURATION
|
|
||||||
|
|
||||||
|
|
||||||
class Integration:
|
class Integration:
|
||||||
@@ -39,7 +37,6 @@ class Integration:
|
|||||||
self.ignored_recipes = []
|
self.ignored_recipes = []
|
||||||
|
|
||||||
description = f'Imported by {request.user.get_user_display_name()} at {date_format(datetime.datetime.now(), "DATETIME_FORMAT")}. Type: {export_type}'
|
description = f'Imported by {request.user.get_user_display_name()} at {date_format(datetime.datetime.now(), "DATETIME_FORMAT")}. Type: {export_type}'
|
||||||
icon = '📥'
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
last_kw = Keyword.objects.filter(name__regex=r'^(Import [0-9]+)', space=request.space).latest('created_at')
|
last_kw = Keyword.objects.filter(name__regex=r'^(Import [0-9]+)', space=request.space).latest('created_at')
|
||||||
@@ -52,23 +49,19 @@ class Integration:
|
|||||||
self.keyword = parent.add_child(
|
self.keyword = parent.add_child(
|
||||||
name=name,
|
name=name,
|
||||||
description=description,
|
description=description,
|
||||||
icon=icon,
|
|
||||||
space=request.space
|
space=request.space
|
||||||
)
|
)
|
||||||
except (IntegrityError, ValueError): # in case, for whatever reason, the name does exist append UUID to it. Not nice but works for now.
|
except (IntegrityError, ValueError): # in case, for whatever reason, the name does exist append UUID to it. Not nice but works for now.
|
||||||
self.keyword = parent.add_child(
|
self.keyword = parent.add_child(
|
||||||
name=f'{name} {str(uuid.uuid4())[0:8]}',
|
name=f'{name} {str(uuid.uuid4())[0:8]}',
|
||||||
description=description,
|
description=description,
|
||||||
icon=icon,
|
|
||||||
space=request.space
|
space=request.space
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def do_export(self, recipes, el):
|
def do_export(self, recipes, el):
|
||||||
|
|
||||||
with scope(space=self.request.space):
|
with scope(space=self.request.space):
|
||||||
el.total_recipes = len(recipes)
|
el.total_recipes = len(recipes)
|
||||||
el.cache_duration = EXPORT_FILE_CACHE_DURATION
|
el.cache_duration = EXPORT_FILE_CACHE_DURATION
|
||||||
el.save()
|
el.save()
|
||||||
|
|
||||||
@@ -80,7 +73,7 @@ class Integration:
|
|||||||
export_file = file
|
export_file = file
|
||||||
|
|
||||||
else:
|
else:
|
||||||
#zip the files if there is more then one file
|
# zip the files if there is more then one file
|
||||||
export_filename = self.get_export_file_name()
|
export_filename = self.get_export_file_name()
|
||||||
export_stream = BytesIO()
|
export_stream = BytesIO()
|
||||||
export_obj = ZipFile(export_stream, 'w')
|
export_obj = ZipFile(export_stream, 'w')
|
||||||
@@ -91,8 +84,7 @@ class Integration:
|
|||||||
export_obj.close()
|
export_obj.close()
|
||||||
export_file = export_stream.getvalue()
|
export_file = export_stream.getvalue()
|
||||||
|
|
||||||
|
cache.set('export_file_' + str(el.pk), {'filename': export_filename, 'file': export_file}, EXPORT_FILE_CACHE_DURATION)
|
||||||
cache.set('export_file_'+str(el.pk), {'filename': export_filename, 'file': export_file}, EXPORT_FILE_CACHE_DURATION)
|
|
||||||
el.running = False
|
el.running = False
|
||||||
el.save()
|
el.save()
|
||||||
|
|
||||||
@@ -100,7 +92,6 @@ class Integration:
|
|||||||
response['Content-Disposition'] = 'attachment; filename="' + export_filename + '"'
|
response['Content-Disposition'] = 'attachment; filename="' + export_filename + '"'
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|
||||||
def import_file_name_filter(self, zip_info_object):
|
def import_file_name_filter(self, zip_info_object):
|
||||||
"""
|
"""
|
||||||
Since zipfile.namelist() returns all files in all subdirectories this function allows filtering of files
|
Since zipfile.namelist() returns all files in all subdirectories this function allows filtering of files
|
||||||
@@ -164,7 +155,7 @@ class Integration:
|
|||||||
|
|
||||||
for z in file_list:
|
for z in file_list:
|
||||||
try:
|
try:
|
||||||
if not hasattr(z, 'filename') or type(z) == Tag:
|
if not hasattr(z, 'filename') or isinstance(z, Tag):
|
||||||
recipe = self.get_recipe_from_file(z)
|
recipe = self.get_recipe_from_file(z)
|
||||||
else:
|
else:
|
||||||
recipe = self.get_recipe_from_file(BytesIO(import_zip.read(z.filename)))
|
recipe = self.get_recipe_from_file(BytesIO(import_zip.read(z.filename)))
|
||||||
@@ -298,7 +289,6 @@ class Integration:
|
|||||||
if DEBUG:
|
if DEBUG:
|
||||||
traceback.print_exc()
|
traceback.print_exc()
|
||||||
|
|
||||||
|
|
||||||
def get_export_file_name(self, format='zip'):
|
def get_export_file_name(self, format='zip'):
|
||||||
return "export_{}.{}".format(datetime.datetime.now().strftime("%Y-%m-%d"), format)
|
return "export_{}.{}".format(datetime.datetime.now().strftime("%Y-%m-%d"), format)
|
||||||
|
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ class Mealie(Integration):
|
|||||||
created_by=self.request.user, internal=True, space=self.request.space)
|
created_by=self.request.user, internal=True, space=self.request.space)
|
||||||
|
|
||||||
for s in recipe_json['recipe_instructions']:
|
for s in recipe_json['recipe_instructions']:
|
||||||
step = Step.objects.create(instruction=s['text'], space=self.request.space, )
|
step = Step.objects.create(instruction=s['text'], space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients, )
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
|
|
||||||
step = recipe.steps.first()
|
step = recipe.steps.first()
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ class MealMaster(Integration):
|
|||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction='\n'.join(directions) + '\n\n', space=self.request.space,
|
instruction='\n'.join(directions) + '\n\n', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ class MelaRecipes(Integration):
|
|||||||
f = ingredient_parser.get_food(food)
|
f = ingredient_parser.get_food(food)
|
||||||
u = ingredient_parser.get_unit(unit)
|
u = ingredient_parser.get_unit(unit)
|
||||||
step.ingredients.add(Ingredient.objects.create(
|
step.ingredients.add(Ingredient.objects.create(
|
||||||
food=f, unit=u, amount=amount, note=note, original_text=ingredient, space=self.request.space,
|
food=f, unit=u, amount=amount, note=note, original_text=ingredient, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
))
|
))
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
|
|
||||||
|
|||||||
@@ -2,13 +2,14 @@ import json
|
|||||||
import re
|
import re
|
||||||
from io import BytesIO, StringIO
|
from io import BytesIO, StringIO
|
||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
|
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
|
|
||||||
from cookbook.helper.image_processing import get_filetype
|
from cookbook.helper.image_processing import get_filetype
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.helper.recipe_url_import import iso_duration_to_minutes
|
from cookbook.helper.recipe_url_import import iso_duration_to_minutes
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Keyword, Recipe, Step, NutritionInformation
|
from cookbook.models import Ingredient, Keyword, NutritionInformation, Recipe, Step
|
||||||
|
|
||||||
|
|
||||||
class NextcloudCookbook(Integration):
|
class NextcloudCookbook(Integration):
|
||||||
@@ -51,9 +52,14 @@ class NextcloudCookbook(Integration):
|
|||||||
|
|
||||||
ingredients_added = False
|
ingredients_added = False
|
||||||
for s in recipe_json['recipeInstructions']:
|
for s in recipe_json['recipeInstructions']:
|
||||||
step = Step.objects.create(
|
if 'text' in s:
|
||||||
instruction=s, space=self.request.space,
|
step = Step.objects.create(
|
||||||
)
|
instruction=s['text'], name=s['name'], space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
step = Step.objects.create(
|
||||||
|
instruction=s, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
|
)
|
||||||
if not ingredients_added:
|
if not ingredients_added:
|
||||||
if len(recipe_json['description'].strip()) > 500:
|
if len(recipe_json['description'].strip()) > 500:
|
||||||
step.instruction = recipe_json['description'].strip() + '\n\n' + step.instruction
|
step.instruction = recipe_json['description'].strip() + '\n\n' + step.instruction
|
||||||
@@ -85,7 +91,7 @@ class NextcloudCookbook(Integration):
|
|||||||
if nutrition != {}:
|
if nutrition != {}:
|
||||||
recipe.nutrition = NutritionInformation.objects.create(**nutrition, space=self.request.space)
|
recipe.nutrition = NutritionInformation.objects.create(**nutrition, space=self.request.space)
|
||||||
recipe.save()
|
recipe.save()
|
||||||
except Exception as e:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
for f in self.files:
|
for f in self.files:
|
||||||
@@ -98,11 +104,10 @@ class NextcloudCookbook(Integration):
|
|||||||
return recipe
|
return recipe
|
||||||
|
|
||||||
def formatTime(self, min):
|
def formatTime(self, min):
|
||||||
h = min//60
|
h = min // 60
|
||||||
m = min % 60
|
m = min % 60
|
||||||
return f'PT{h}H{m}M0S'
|
return f'PT{h}H{m}M0S'
|
||||||
|
|
||||||
|
|
||||||
def get_file_from_recipe(self, recipe):
|
def get_file_from_recipe(self, recipe):
|
||||||
|
|
||||||
export = {}
|
export = {}
|
||||||
@@ -111,7 +116,7 @@ class NextcloudCookbook(Integration):
|
|||||||
export['url'] = recipe.source_url
|
export['url'] = recipe.source_url
|
||||||
export['prepTime'] = self.formatTime(recipe.working_time)
|
export['prepTime'] = self.formatTime(recipe.working_time)
|
||||||
export['cookTime'] = self.formatTime(recipe.waiting_time)
|
export['cookTime'] = self.formatTime(recipe.waiting_time)
|
||||||
export['totalTime'] = self.formatTime(recipe.working_time+recipe.waiting_time)
|
export['totalTime'] = self.formatTime(recipe.working_time + recipe.waiting_time)
|
||||||
export['recipeYield'] = recipe.servings
|
export['recipeYield'] = recipe.servings
|
||||||
export['image'] = f'/Recipes/{recipe.name}/full.jpg'
|
export['image'] = f'/Recipes/{recipe.name}/full.jpg'
|
||||||
export['imageUrl'] = f'/Recipes/{recipe.name}/full.jpg'
|
export['imageUrl'] = f'/Recipes/{recipe.name}/full.jpg'
|
||||||
@@ -133,7 +138,6 @@ class NextcloudCookbook(Integration):
|
|||||||
export['recipeIngredient'] = recipeIngredient
|
export['recipeIngredient'] = recipeIngredient
|
||||||
export['recipeInstructions'] = recipeInstructions
|
export['recipeInstructions'] = recipeInstructions
|
||||||
|
|
||||||
|
|
||||||
return "recipe.json", json.dumps(export)
|
return "recipe.json", json.dumps(export)
|
||||||
|
|
||||||
def get_files_from_recipes(self, recipes, el, cookie):
|
def get_files_from_recipes(self, recipes, el, cookie):
|
||||||
@@ -163,7 +167,7 @@ class NextcloudCookbook(Integration):
|
|||||||
|
|
||||||
export_zip_obj.close()
|
export_zip_obj.close()
|
||||||
|
|
||||||
return [[ self.get_export_file_name(), export_zip_stream.getvalue() ]]
|
return [[self.get_export_file_name(), export_zip_stream.getvalue()]]
|
||||||
|
|
||||||
def getJPEG(self, imageByte):
|
def getJPEG(self, imageByte):
|
||||||
image = Image.open(BytesIO(imageByte))
|
image = Image.open(BytesIO(imageByte))
|
||||||
@@ -172,14 +176,14 @@ class NextcloudCookbook(Integration):
|
|||||||
bytes = BytesIO()
|
bytes = BytesIO()
|
||||||
image.save(bytes, "JPEG")
|
image.save(bytes, "JPEG")
|
||||||
return bytes.getvalue()
|
return bytes.getvalue()
|
||||||
|
|
||||||
def getThumb(self, size, imageByte):
|
def getThumb(self, size, imageByte):
|
||||||
image = Image.open(BytesIO(imageByte))
|
image = Image.open(BytesIO(imageByte))
|
||||||
|
|
||||||
w, h = image.size
|
w, h = image.size
|
||||||
m = min(w, h)
|
m = min(w, h)
|
||||||
|
|
||||||
image = image.crop(((w-m)//2, (h-m)//2, (w+m)//2, (h+m)//2))
|
image = image.crop(((w - m) // 2, (h - m) // 2, (w + m) // 2, (h + m) // 2))
|
||||||
image = image.resize([size, size], Image.Resampling.LANCZOS)
|
image = image.resize([size, size], Image.Resampling.LANCZOS)
|
||||||
image = image.convert('RGB')
|
image = image.convert('RGB')
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
import json
|
import json
|
||||||
|
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Recipe, Step, Keyword, Comment, CookLog
|
from cookbook.models import Comment, CookLog, Ingredient, Keyword, Recipe, Step
|
||||||
from django.utils.translation import gettext as _
|
|
||||||
|
|
||||||
class OpenEats(Integration):
|
class OpenEats(Integration):
|
||||||
|
|
||||||
@@ -25,16 +27,16 @@ class OpenEats(Integration):
|
|||||||
if file["source"] != '':
|
if file["source"] != '':
|
||||||
instructions += '\n' + _('Recipe source:') + f'[{file["source"]}]({file["source"]})'
|
instructions += '\n' + _('Recipe source:') + f'[{file["source"]}]({file["source"]})'
|
||||||
|
|
||||||
cuisine_keyword, created = Keyword.objects.get_or_create(name="Cuisine", space=self.request.space)
|
cuisine_keyword, created = Keyword.objects.get_or_create(name="Cuisine", space=self.request.space)
|
||||||
if file["cuisine"] != '':
|
if file["cuisine"] != '':
|
||||||
keyword, created = Keyword.objects.get_or_create(name=file["cuisine"].strip(), space=self.request.space)
|
keyword, created = Keyword.objects.get_or_create(name=file["cuisine"].strip(), space=self.request.space)
|
||||||
if created:
|
if created:
|
||||||
keyword.move(cuisine_keyword, pos="last-child")
|
keyword.move(cuisine_keyword, pos="last-child")
|
||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
|
|
||||||
course_keyword, created = Keyword.objects.get_or_create(name="Course", space=self.request.space)
|
course_keyword, created = Keyword.objects.get_or_create(name="Course", space=self.request.space)
|
||||||
if file["course"] != '':
|
if file["course"] != '':
|
||||||
keyword, created = Keyword.objects.get_or_create(name=file["course"].strip(), space=self.request.space)
|
keyword, created = Keyword.objects.get_or_create(name=file["course"].strip(), space=self.request.space)
|
||||||
if created:
|
if created:
|
||||||
keyword.move(course_keyword, pos="last-child")
|
keyword.move(course_keyword, pos="last-child")
|
||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
@@ -51,7 +53,7 @@ class OpenEats(Integration):
|
|||||||
recipe.image = f'recipes/openeats-import/{file["photo"]}'
|
recipe.image = f'recipes/openeats-import/{file["photo"]}'
|
||||||
recipe.save()
|
recipe.save()
|
||||||
|
|
||||||
step = Step.objects.create(instruction=instructions, space=self.request.space,)
|
step = Step.objects.create(instruction=instructions, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,)
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
for ingredient in file['ingredients']:
|
for ingredient in file['ingredients']:
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ class Paprika(Integration):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction=instructions, space=self.request.space,
|
instruction=instructions, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
if 'description' in recipe_json and len(recipe_json['description'].strip()) > 500:
|
if 'description' in recipe_json and len(recipe_json['description'].strip()) > 500:
|
||||||
@@ -90,7 +90,7 @@ class Paprika(Integration):
|
|||||||
if validators.url(url, public=True):
|
if validators.url(url, public=True):
|
||||||
response = requests.get(url)
|
response = requests.get(url)
|
||||||
self.import_recipe_image(recipe, BytesIO(response.content))
|
self.import_recipe_image(recipe, BytesIO(response.content))
|
||||||
except:
|
except Exception:
|
||||||
if recipe_json.get("photo_data", None):
|
if recipe_json.get("photo_data", None):
|
||||||
self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_json['photo_data'])), filetype='.jpeg')
|
self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_json['photo_data'])), filetype='.jpeg')
|
||||||
|
|
||||||
|
|||||||
@@ -1,21 +1,11 @@
|
|||||||
import json
|
|
||||||
from io import BytesIO
|
|
||||||
from re import match
|
|
||||||
from zipfile import ZipFile
|
|
||||||
import asyncio
|
import asyncio
|
||||||
from pyppeteer import launch
|
|
||||||
|
|
||||||
from rest_framework.renderers import JSONRenderer
|
|
||||||
|
|
||||||
from cookbook.helper.image_processing import get_filetype
|
|
||||||
from cookbook.integration.integration import Integration
|
|
||||||
from cookbook.serializer import RecipeExportSerializer
|
|
||||||
|
|
||||||
from cookbook.models import ExportLog
|
|
||||||
from asgiref.sync import sync_to_async
|
|
||||||
|
|
||||||
import django.core.management.commands.runserver as runserver
|
import django.core.management.commands.runserver as runserver
|
||||||
import logging
|
from asgiref.sync import sync_to_async
|
||||||
|
from pyppeteer import launch
|
||||||
|
|
||||||
|
from cookbook.integration.integration import Integration
|
||||||
|
|
||||||
|
|
||||||
class PDFexport(Integration):
|
class PDFexport(Integration):
|
||||||
|
|
||||||
@@ -42,7 +32,6 @@ class PDFexport(Integration):
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
files = []
|
files = []
|
||||||
for recipe in recipes:
|
for recipe in recipes:
|
||||||
|
|
||||||
@@ -50,20 +39,18 @@ class PDFexport(Integration):
|
|||||||
await page.emulateMedia('print')
|
await page.emulateMedia('print')
|
||||||
await page.setCookie(cookies)
|
await page.setCookie(cookies)
|
||||||
|
|
||||||
await page.goto('http://'+cmd.default_addr+':'+cmd.default_port+'/view/recipe/'+str(recipe.id), {'waitUntil': 'domcontentloaded'})
|
await page.goto('http://' + cmd.default_addr + ':' + cmd.default_port + '/view/recipe/' + str(recipe.id), {'waitUntil': 'domcontentloaded'})
|
||||||
await page.waitForSelector('#printReady');
|
await page.waitForSelector('#printReady')
|
||||||
|
|
||||||
files.append([recipe.name + '.pdf', await page.pdf(options)])
|
files.append([recipe.name + '.pdf', await page.pdf(options)])
|
||||||
await page.close();
|
await page.close()
|
||||||
|
|
||||||
el.exported_recipes += 1
|
el.exported_recipes += 1
|
||||||
el.msg += self.get_recipe_processed_msg(recipe)
|
el.msg += self.get_recipe_processed_msg(recipe)
|
||||||
await sync_to_async(el.save, thread_sensitive=True)()
|
await sync_to_async(el.save, thread_sensitive=True)()
|
||||||
|
|
||||||
|
|
||||||
await browser.close()
|
await browser.close()
|
||||||
return files
|
return files
|
||||||
|
|
||||||
|
|
||||||
def get_files_from_recipes(self, recipes, el, cookie):
|
def get_files_from_recipes(self, recipes, el, cookie):
|
||||||
return asyncio.run(self.get_files_from_recipes_async(recipes, el, cookie))
|
return asyncio.run(self.get_files_from_recipes_async(recipes, el, cookie))
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ class Pepperplate(Integration):
|
|||||||
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space)
|
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space)
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction='\n'.join(directions) + '\n\n', space=self.request.space,
|
instruction='\n'.join(directions) + '\n\n', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
import validators
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
@@ -45,7 +46,7 @@ class Plantoeat(Integration):
|
|||||||
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space)
|
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space)
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction='\n'.join(directions) + '\n\n', space=self.request.space,
|
instruction='\n'.join(directions) + '\n\n', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
if tags:
|
if tags:
|
||||||
@@ -67,8 +68,9 @@ class Plantoeat(Integration):
|
|||||||
|
|
||||||
if image_url:
|
if image_url:
|
||||||
try:
|
try:
|
||||||
response = requests.get(image_url)
|
if validators.url(image_url, public=True):
|
||||||
self.import_recipe_image(recipe, BytesIO(response.content))
|
response = requests.get(image_url)
|
||||||
|
self.import_recipe_image(recipe, BytesIO(response.content))
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print('failed to import image ', str(e))
|
print('failed to import image ', str(e))
|
||||||
|
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ class RecetteTek(Integration):
|
|||||||
if not instructions:
|
if not instructions:
|
||||||
instructions = ''
|
instructions = ''
|
||||||
|
|
||||||
step = Step.objects.create(instruction=instructions, space=self.request.space,)
|
step = Step.objects.create(instruction=instructions, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,)
|
||||||
|
|
||||||
# Append the original import url to the step (if it exists)
|
# Append the original import url to the step (if it exists)
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ class RecipeKeeper(Integration):
|
|||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
step = Step.objects.create(instruction='', space=self.request.space, )
|
step = Step.objects.create(instruction='', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients, )
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
for ingredient in file.find("div", {"itemprop": "recipeIngredients"}).findChildren("p"):
|
for ingredient in file.find("div", {"itemprop": "recipeIngredients"}).findChildren("p"):
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import requests
|
|||||||
import validators
|
import validators
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
|
from cookbook.helper.recipe_url_import import parse_servings, parse_servings_text, parse_time
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Recipe, Step
|
from cookbook.models import Ingredient, Recipe, Step
|
||||||
|
|
||||||
@@ -18,19 +19,21 @@ class RecipeSage(Integration):
|
|||||||
created_by=self.request.user, internal=True,
|
created_by=self.request.user, internal=True,
|
||||||
space=self.request.space)
|
space=self.request.space)
|
||||||
|
|
||||||
|
if file['recipeYield'] != '':
|
||||||
|
recipe.servings = parse_servings(file['recipeYield'])
|
||||||
|
recipe.servings_text = parse_servings_text(file['recipeYield'])
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if file['recipeYield'] != '':
|
if 'totalTime' in file and file['totalTime'] != '':
|
||||||
recipe.servings = int(file['recipeYield'])
|
recipe.working_time = parse_time(file['totalTime'])
|
||||||
|
|
||||||
if file['totalTime'] != '':
|
if 'timePrep' in file and file['prepTime'] != '':
|
||||||
recipe.waiting_time = int(file['totalTime']) - int(file['timePrep'])
|
recipe.working_time = parse_time(file['timePrep'])
|
||||||
|
recipe.waiting_time = parse_time(file['totalTime']) - parse_time(file['timePrep'])
|
||||||
if file['prepTime'] != '':
|
|
||||||
recipe.working_time = int(file['timePrep'])
|
|
||||||
|
|
||||||
recipe.save()
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print('failed to parse yield or time ', str(e))
|
print('failed to parse time ', str(e))
|
||||||
|
|
||||||
|
recipe.save()
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
ingredients_added = False
|
ingredients_added = False
|
||||||
@@ -46,7 +49,7 @@ class RecipeSage(Integration):
|
|||||||
f = ingredient_parser.get_food(food)
|
f = ingredient_parser.get_food(food)
|
||||||
u = ingredient_parser.get_unit(unit)
|
u = ingredient_parser.get_unit(unit)
|
||||||
step.ingredients.add(Ingredient.objects.create(
|
step.ingredients.add(Ingredient.objects.create(
|
||||||
food=f, unit=u, amount=amount, note=note, original_text=ingredient, space=self.request.space,
|
food=f, unit=u, amount=amount, note=note, original_text=ingredient, space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
))
|
))
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
|
|
||||||
|
|||||||
@@ -2,12 +2,10 @@ import base64
|
|||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
from xml import etree
|
from xml import etree
|
||||||
|
|
||||||
from lxml import etree
|
|
||||||
|
|
||||||
from cookbook.helper.ingredient_parser import IngredientParser
|
from cookbook.helper.ingredient_parser import IngredientParser
|
||||||
from cookbook.helper.recipe_url_import import parse_time, parse_servings, parse_servings_text
|
from cookbook.helper.recipe_url_import import parse_servings, parse_servings_text
|
||||||
from cookbook.integration.integration import Integration
|
from cookbook.integration.integration import Integration
|
||||||
from cookbook.models import Ingredient, Recipe, Step, Keyword
|
from cookbook.models import Ingredient, Keyword, Recipe, Step
|
||||||
|
|
||||||
|
|
||||||
class Rezeptsuitede(Integration):
|
class Rezeptsuitede(Integration):
|
||||||
@@ -22,9 +20,12 @@ class Rezeptsuitede(Integration):
|
|||||||
name=recipe_xml.find('head').attrib['title'].strip(),
|
name=recipe_xml.find('head').attrib['title'].strip(),
|
||||||
created_by=self.request.user, internal=True, space=self.request.space)
|
created_by=self.request.user, internal=True, space=self.request.space)
|
||||||
|
|
||||||
if recipe_xml.find('head').attrib['servingtype']:
|
try:
|
||||||
recipe.servings = parse_servings(recipe_xml.find('head').attrib['servingtype'].strip())
|
if recipe_xml.find('head').attrib['servingtype']:
|
||||||
recipe.servings_text = parse_servings_text(recipe_xml.find('head').attrib['servingtype'].strip())
|
recipe.servings = parse_servings(recipe_xml.find('head').attrib['servingtype'].strip())
|
||||||
|
recipe.servings_text = parse_servings_text(recipe_xml.find('head').attrib['servingtype'].strip())
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
|
||||||
if recipe_xml.find('remark') is not None: # description is a list of <li>'s with text
|
if recipe_xml.find('remark') is not None: # description is a list of <li>'s with text
|
||||||
if recipe_xml.find('remark').find('line') is not None:
|
if recipe_xml.find('remark').find('line') is not None:
|
||||||
@@ -34,7 +35,7 @@ class Rezeptsuitede(Integration):
|
|||||||
try:
|
try:
|
||||||
if prep.find('step').text:
|
if prep.find('step').text:
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction=prep.find('step').text.strip(), space=self.request.space,
|
instruction=prep.find('step').text.strip(), space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
recipe.steps.add(step)
|
recipe.steps.add(step)
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -50,20 +51,22 @@ class Rezeptsuitede(Integration):
|
|||||||
for ingredient in recipe_xml.find('part').findall('ingredient'):
|
for ingredient in recipe_xml.find('part').findall('ingredient'):
|
||||||
f = ingredient_parser.get_food(ingredient.attrib['item'])
|
f = ingredient_parser.get_food(ingredient.attrib['item'])
|
||||||
u = ingredient_parser.get_unit(ingredient.attrib['unit'])
|
u = ingredient_parser.get_unit(ingredient.attrib['unit'])
|
||||||
amount, unit, note = ingredient_parser.parse_amount(ingredient.attrib['qty'])
|
amount = 0
|
||||||
|
if ingredient.attrib['qty'].strip() != '':
|
||||||
|
amount, unit, note = ingredient_parser.parse_amount(ingredient.attrib['qty'])
|
||||||
ingredient_step.ingredients.add(Ingredient.objects.create(food=f, unit=u, amount=amount, space=self.request.space, ))
|
ingredient_step.ingredients.add(Ingredient.objects.create(food=f, unit=u, amount=amount, space=self.request.space, ))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
k, created = Keyword.objects.get_or_create(name=recipe_xml.find('head').find('cat').text.strip(), space=self.request.space)
|
k, created = Keyword.objects.get_or_create(name=recipe_xml.find('head').find('cat').text.strip(), space=self.request.space)
|
||||||
recipe.keywords.add(k)
|
recipe.keywords.add(k)
|
||||||
except Exception as e:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
recipe.save()
|
recipe.save()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_xml.find('head').find('picbin').text)), filetype='.jpeg')
|
self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_xml.find('head').find('picbin').text)), filetype='.jpeg')
|
||||||
except:
|
except BaseException:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
return recipe
|
return recipe
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ class RezKonv(Integration):
|
|||||||
recipe.keywords.add(keyword)
|
recipe.keywords.add(keyword)
|
||||||
|
|
||||||
step = Step.objects.create(
|
step = Step.objects.create(
|
||||||
instruction=' \n'.join(directions) + '\n\n', space=self.request.space,
|
instruction=' \n'.join(directions) + '\n\n', space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients,
|
||||||
)
|
)
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
@@ -60,8 +60,8 @@ class RezKonv(Integration):
|
|||||||
def split_recipe_file(self, file):
|
def split_recipe_file(self, file):
|
||||||
recipe_list = []
|
recipe_list = []
|
||||||
current_recipe = ''
|
current_recipe = ''
|
||||||
encoding_list = ['windows-1250',
|
# TODO build algorithm to try trough encodings and fail if none work, use for all importers
|
||||||
'latin-1'] # TODO build algorithm to try trough encodings and fail if none work, use for all importers
|
# encoding_list = ['windows-1250', 'latin-1']
|
||||||
encoding = 'windows-1250'
|
encoding = 'windows-1250'
|
||||||
for fl in file.readlines():
|
for fl in file.readlines():
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ class Saffron(Integration):
|
|||||||
|
|
||||||
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space, )
|
recipe = Recipe.objects.create(name=title, description=description, created_by=self.request.user, internal=True, space=self.request.space, )
|
||||||
|
|
||||||
step = Step.objects.create(instruction='\n'.join(directions), space=self.request.space, )
|
step = Step.objects.create(instruction='\n'.join(directions), space=self.request.space, show_ingredients_table=self.request.user.userpreference.show_step_ingredients, )
|
||||||
|
|
||||||
ingredient_parser = IngredientParser(self.request, True)
|
ingredient_parser = IngredientParser(self.request, True)
|
||||||
for ingredient in ingredients:
|
for ingredient in ingredients:
|
||||||
@@ -59,11 +59,11 @@ class Saffron(Integration):
|
|||||||
|
|
||||||
def get_file_from_recipe(self, recipe):
|
def get_file_from_recipe(self, recipe):
|
||||||
|
|
||||||
data = "Title: "+recipe.name if recipe.name else ""+"\n"
|
data = "Title: " + recipe.name if recipe.name else "" + "\n"
|
||||||
data += "Description: "+recipe.description if recipe.description else ""+"\n"
|
data += "Description: " + recipe.description if recipe.description else "" + "\n"
|
||||||
data += "Source: \n"
|
data += "Source: \n"
|
||||||
data += "Original URL: \n"
|
data += "Original URL: \n"
|
||||||
data += "Yield: "+str(recipe.servings)+"\n"
|
data += "Yield: " + str(recipe.servings) + "\n"
|
||||||
data += "Cookbook: \n"
|
data += "Cookbook: \n"
|
||||||
data += "Section: \n"
|
data += "Section: \n"
|
||||||
data += "Image: \n"
|
data += "Image: \n"
|
||||||
@@ -78,13 +78,13 @@ class Saffron(Integration):
|
|||||||
|
|
||||||
data += "Ingredients: \n"
|
data += "Ingredients: \n"
|
||||||
for ingredient in recipeIngredient:
|
for ingredient in recipeIngredient:
|
||||||
data += ingredient+"\n"
|
data += ingredient + "\n"
|
||||||
|
|
||||||
data += "Instructions: \n"
|
data += "Instructions: \n"
|
||||||
for instruction in recipeInstructions:
|
for instruction in recipeInstructions:
|
||||||
data += instruction+"\n"
|
data += instruction + "\n"
|
||||||
|
|
||||||
return recipe.name+'.txt', data
|
return recipe.name + '.txt', data
|
||||||
|
|
||||||
def get_files_from_recipes(self, recipes, el, cookie):
|
def get_files_from_recipes(self, recipes, el, cookie):
|
||||||
files = []
|
files = []
|
||||||
|
|||||||
Binary file not shown.
@@ -8,8 +8,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
||||||
"PO-Revision-Date: 2022-05-10 15:32+0000\n"
|
"PO-Revision-Date: 2023-04-12 11:55+0000\n"
|
||||||
"Last-Translator: zeon <zeonbg@gmail.com>\n"
|
"Last-Translator: noxonad <noxonad@proton.me>\n"
|
||||||
"Language-Team: Bulgarian <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Bulgarian <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/bg/>\n"
|
"recipes-backend/bg/>\n"
|
||||||
"Language: bg\n"
|
"Language: bg\n"
|
||||||
@@ -17,7 +17,7 @@ msgstr ""
|
|||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
||||||
"X-Generator: Weblate 4.10.1\n"
|
"X-Generator: Weblate 4.15\n"
|
||||||
|
|
||||||
#: .\cookbook\filters.py:23 .\cookbook\templates\forms\ingredients.html:34
|
#: .\cookbook\filters.py:23 .\cookbook\templates\forms\ingredients.html:34
|
||||||
#: .\cookbook\templates\space.html:49 .\cookbook\templates\stats.html:28
|
#: .\cookbook\templates\space.html:49 .\cookbook\templates\stats.html:28
|
||||||
@@ -1433,7 +1433,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\index.html:29
|
#: .\cookbook\templates\index.html:29
|
||||||
msgid "Search recipe ..."
|
msgid "Search recipe ..."
|
||||||
msgstr "Търсете рецепта..."
|
msgstr "Търсете рецепта ..."
|
||||||
|
|
||||||
#: .\cookbook\templates\index.html:44
|
#: .\cookbook\templates\index.html:44
|
||||||
msgid "New Recipe"
|
msgid "New Recipe"
|
||||||
@@ -1818,7 +1818,7 @@ msgid ""
|
|||||||
msgstr ""
|
msgstr ""
|
||||||
" \n"
|
" \n"
|
||||||
" Пълнотекстови търсения се опитват да нормализират предоставените "
|
" Пълнотекстови търсения се опитват да нормализират предоставените "
|
||||||
"думи, за да съответстват на често срещани варианти. Например: 'вили, "
|
"думи, за да съответстват на често срещани варианти. Например: 'вили, "
|
||||||
"'вилица', 'вилици' всички ще се нормализират до 'вилиц'.\n"
|
"'вилица', 'вилици' всички ще се нормализират до 'вилиц'.\n"
|
||||||
" Има няколко налични метода, описани по-долу, които ще "
|
" Има няколко налични метода, описани по-долу, които ще "
|
||||||
"контролират как поведението при търсене трябва да реагира, когато се търсят "
|
"контролират как поведението при търсене трябва да реагира, когато се търсят "
|
||||||
|
|||||||
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -11,8 +11,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2021-02-09 18:01+0100\n"
|
"POT-Creation-Date: 2021-02-09 18:01+0100\n"
|
||||||
"PO-Revision-Date: 2023-03-25 11:32+0000\n"
|
"PO-Revision-Date: 2023-07-31 14:19+0000\n"
|
||||||
"Last-Translator: Matěj Kubla <matykubla@gmail.com>\n"
|
"Last-Translator: Mára Štěpánek <stepanekm7@gmail.com>\n"
|
||||||
"Language-Team: Czech <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Czech <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/cs/>\n"
|
"recipes-backend/cs/>\n"
|
||||||
"Language: cs\n"
|
"Language: cs\n"
|
||||||
@@ -36,7 +36,7 @@ msgid ""
|
|||||||
"try them out!"
|
"try them out!"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Barva horního navigačního menu. Některé barvy neladí se všemi tématy a je "
|
"Barva horního navigačního menu. Některé barvy neladí se všemi tématy a je "
|
||||||
"třeba je vyzkoušet."
|
"třeba je vyzkoušet!"
|
||||||
|
|
||||||
#: .\cookbook\forms.py:45
|
#: .\cookbook\forms.py:45
|
||||||
msgid "Default Unit to be used when inserting a new ingredient into a recipe."
|
msgid "Default Unit to be used when inserting a new ingredient into a recipe."
|
||||||
@@ -50,7 +50,7 @@ msgid ""
|
|||||||
"to fractions automatically)"
|
"to fractions automatically)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Povolit podporu zlomků u množství ingrediencí (desetinná čísla budou "
|
"Povolit podporu zlomků u množství ingrediencí (desetinná čísla budou "
|
||||||
"automaticky převedena na zlomky)."
|
"automaticky převedena na zlomky)"
|
||||||
|
|
||||||
#: .\cookbook\forms.py:47
|
#: .\cookbook\forms.py:47
|
||||||
msgid ""
|
msgid ""
|
||||||
|
|||||||
Binary file not shown.
@@ -8,8 +8,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
||||||
"PO-Revision-Date: 2023-03-06 10:55+0000\n"
|
"PO-Revision-Date: 2023-04-12 11:55+0000\n"
|
||||||
"Last-Translator: Anders Obro <oebro@duck.com>\n"
|
"Last-Translator: noxonad <noxonad@proton.me>\n"
|
||||||
"Language-Team: Danish <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Danish <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/da/>\n"
|
"recipes-backend/da/>\n"
|
||||||
"Language: da\n"
|
"Language: da\n"
|
||||||
@@ -1806,7 +1806,7 @@ msgid ""
|
|||||||
msgstr ""
|
msgstr ""
|
||||||
" \n"
|
" \n"
|
||||||
" Heltekstsøgning forsøger at normalisere de givne ord så de "
|
" Heltekstsøgning forsøger at normalisere de givne ord så de "
|
||||||
"matcher stammevarianter. F.eks: 'skeen', 'skeer' og 'sket' vil alt "
|
"matcher stammevarianter. F.eks: 'skeen', 'skeer' og 'sket' vil alt "
|
||||||
"normaliseres til 'ske'.\n"
|
"normaliseres til 'ske'.\n"
|
||||||
" Der er flere metoder tilgængelige, beskrevet herunder, som vil "
|
" Der er flere metoder tilgængelige, beskrevet herunder, som vil "
|
||||||
"bestemme hvordan søgningen skal opfører sig når flere søgeord er angivet.\n"
|
"bestemme hvordan søgningen skal opfører sig når flere søgeord er angivet.\n"
|
||||||
|
|||||||
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
Binary file not shown.
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -8,8 +8,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2022-02-11 08:52+0100\n"
|
"POT-Creation-Date: 2022-02-11 08:52+0100\n"
|
||||||
"PO-Revision-Date: 2023-02-18 10:55+0000\n"
|
"PO-Revision-Date: 2023-04-12 11:55+0000\n"
|
||||||
"Last-Translator: Joachim Weber <joachim.weber@gmx.de>\n"
|
"Last-Translator: noxonad <noxonad@proton.me>\n"
|
||||||
"Language-Team: Portuguese (Brazil) <http://translate.tandoor.dev/projects/"
|
"Language-Team: Portuguese (Brazil) <http://translate.tandoor.dev/projects/"
|
||||||
"tandoor/recipes-backend/pt_BR/>\n"
|
"tandoor/recipes-backend/pt_BR/>\n"
|
||||||
"Language: pt_BR\n"
|
"Language: pt_BR\n"
|
||||||
@@ -2208,7 +2208,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\url_import.html:38
|
#: .\cookbook\templates\url_import.html:38
|
||||||
msgid "URL"
|
msgid "URL"
|
||||||
msgstr ""
|
msgstr "URL"
|
||||||
|
|
||||||
#: .\cookbook\templates\url_import.html:40
|
#: .\cookbook\templates\url_import.html:40
|
||||||
msgid "App"
|
msgid "App"
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -8,17 +8,17 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2021-09-13 22:40+0200\n"
|
"POT-Creation-Date: 2021-09-13 22:40+0200\n"
|
||||||
"PO-Revision-Date: 2022-11-30 19:09+0000\n"
|
"PO-Revision-Date: 2023-05-01 07:55+0000\n"
|
||||||
"Last-Translator: Alex <kovsharoff@gmail.com>\n"
|
"Last-Translator: axeron2036 <admin@axeron2036.ru>\n"
|
||||||
"Language-Team: Russian <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Russian <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/ru/>\n"
|
"recipes-backend/ru/>\n"
|
||||||
"Language: ru\n"
|
"Language: ru\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n"
|
"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && "
|
||||||
"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
|
"n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
|
||||||
"X-Generator: Weblate 4.14.1\n"
|
"X-Generator: Weblate 4.15\n"
|
||||||
|
|
||||||
#: .\cookbook\filters.py:23 .\cookbook\templates\base.html:125
|
#: .\cookbook\filters.py:23 .\cookbook\templates\base.html:125
|
||||||
#: .\cookbook\templates\forms\ingredients.html:34
|
#: .\cookbook\templates\forms\ingredients.html:34
|
||||||
@@ -286,7 +286,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\forms.py:497
|
#: .\cookbook\forms.py:497
|
||||||
msgid "Search Method"
|
msgid "Search Method"
|
||||||
msgstr ""
|
msgstr "Способ поиска"
|
||||||
|
|
||||||
#: .\cookbook\forms.py:498
|
#: .\cookbook\forms.py:498
|
||||||
msgid "Fuzzy Lookups"
|
msgid "Fuzzy Lookups"
|
||||||
@@ -861,7 +861,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\base.html:220
|
#: .\cookbook\templates\base.html:220
|
||||||
msgid "GitHub"
|
msgid "GitHub"
|
||||||
msgstr ""
|
msgstr "GitHub"
|
||||||
|
|
||||||
#: .\cookbook\templates\base.html:224
|
#: .\cookbook\templates\base.html:224
|
||||||
msgid "API Browser"
|
msgid "API Browser"
|
||||||
@@ -1937,7 +1937,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\space.html:106
|
#: .\cookbook\templates\space.html:106
|
||||||
msgid "user"
|
msgid "user"
|
||||||
msgstr ""
|
msgstr "пользователь"
|
||||||
|
|
||||||
#: .\cookbook\templates\space.html:107
|
#: .\cookbook\templates\space.html:107
|
||||||
msgid "guest"
|
msgid "guest"
|
||||||
|
|||||||
Binary file not shown.
@@ -8,17 +8,17 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2021-11-08 16:27+0100\n"
|
"POT-Creation-Date: 2021-11-08 16:27+0100\n"
|
||||||
"PO-Revision-Date: 2022-02-02 15:31+0000\n"
|
"PO-Revision-Date: 2023-08-13 08:19+0000\n"
|
||||||
"Last-Translator: Mario Dvorsek <mario.dvorsek@gmail.com>\n"
|
"Last-Translator: Miha Perpar <miha.perpar2@gmail.com>\n"
|
||||||
"Language-Team: Slovenian <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Slovenian <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/sl/>\n"
|
"recipes-backend/sl/>\n"
|
||||||
"Language: sl\n"
|
"Language: sl\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=4; plural=n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || n"
|
"Plural-Forms: nplurals=4; plural=n%100==1 ? 0 : n%100==2 ? 1 : n%100==3 || "
|
||||||
"%100==4 ? 2 : 3;\n"
|
"n%100==4 ? 2 : 3;\n"
|
||||||
"X-Generator: Weblate 4.10.1\n"
|
"X-Generator: Weblate 4.15\n"
|
||||||
|
|
||||||
#: .\cookbook\filters.py:23 .\cookbook\templates\base.html:125
|
#: .\cookbook\filters.py:23 .\cookbook\templates\base.html:125
|
||||||
#: .\cookbook\templates\forms\ingredients.html:34
|
#: .\cookbook\templates\forms\ingredients.html:34
|
||||||
@@ -964,7 +964,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\base.html:275
|
#: .\cookbook\templates\base.html:275
|
||||||
msgid "GitHub"
|
msgid "GitHub"
|
||||||
msgstr ""
|
msgstr "GitHub"
|
||||||
|
|
||||||
#: .\cookbook\templates\base.html:277
|
#: .\cookbook\templates\base.html:277
|
||||||
msgid "Translate Tandoor"
|
msgid "Translate Tandoor"
|
||||||
@@ -1961,7 +1961,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\space.html:106
|
#: .\cookbook\templates\space.html:106
|
||||||
msgid "user"
|
msgid "user"
|
||||||
msgstr ""
|
msgstr "uporabnik"
|
||||||
|
|
||||||
#: .\cookbook\templates\space.html:107
|
#: .\cookbook\templates\space.html:107
|
||||||
msgid "guest"
|
msgid "guest"
|
||||||
@@ -2107,7 +2107,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\url_import.html:36
|
#: .\cookbook\templates\url_import.html:36
|
||||||
msgid "URL"
|
msgid "URL"
|
||||||
msgstr ""
|
msgstr "URL"
|
||||||
|
|
||||||
#: .\cookbook\templates\url_import.html:38
|
#: .\cookbook\templates\url_import.html:38
|
||||||
msgid "App"
|
msgid "App"
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -8,8 +8,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
"POT-Creation-Date: 2022-04-29 18:42+0200\n"
|
||||||
"PO-Revision-Date: 2023-02-09 13:55+0000\n"
|
"PO-Revision-Date: 2023-04-12 11:55+0000\n"
|
||||||
"Last-Translator: vertilo <vertilo.dev@gmail.com>\n"
|
"Last-Translator: noxonad <noxonad@proton.me>\n"
|
||||||
"Language-Team: Ukrainian <http://translate.tandoor.dev/projects/tandoor/"
|
"Language-Team: Ukrainian <http://translate.tandoor.dev/projects/tandoor/"
|
||||||
"recipes-backend/uk/>\n"
|
"recipes-backend/uk/>\n"
|
||||||
"Language: uk\n"
|
"Language: uk\n"
|
||||||
@@ -1091,7 +1091,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: .\cookbook\templates\base.html:311
|
#: .\cookbook\templates\base.html:311
|
||||||
msgid "GitHub"
|
msgid "GitHub"
|
||||||
msgstr ""
|
msgstr "GitHub"
|
||||||
|
|
||||||
#: .\cookbook\templates\base.html:313
|
#: .\cookbook\templates\base.html:313
|
||||||
msgid "Translate Tandoor"
|
msgid "Translate Tandoor"
|
||||||
|
|||||||
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -34,35 +34,14 @@ class RecipeSearchManager(models.Manager):
|
|||||||
+ SearchVector(StringAgg('steps__ingredients__food__name__unaccent', delimiter=' '), weight='B', config=language)
|
+ SearchVector(StringAgg('steps__ingredients__food__name__unaccent', delimiter=' '), weight='B', config=language)
|
||||||
+ SearchVector(StringAgg('keywords__name__unaccent', delimiter=' '), weight='B', config=language))
|
+ SearchVector(StringAgg('keywords__name__unaccent', delimiter=' '), weight='B', config=language))
|
||||||
search_rank = SearchRank(search_vectors, search_query)
|
search_rank = SearchRank(search_vectors, search_query)
|
||||||
# USING TRIGRAM BREAKS WEB SEARCH
|
|
||||||
# ADDING MULTIPLE TRIGRAMS CREATES DUPLICATE RESULTS
|
|
||||||
# DISTINCT NOT COMPAITBLE WITH ANNOTATE
|
|
||||||
# trigram_name = (TrigramSimilarity('name', search_text))
|
|
||||||
# trigram_description = (TrigramSimilarity('description', search_text))
|
|
||||||
# trigram_food = (TrigramSimilarity('steps__ingredients__food__name', search_text))
|
|
||||||
# trigram_keyword = (TrigramSimilarity('keywords__name', search_text))
|
|
||||||
# adding additional trigrams created duplicates
|
|
||||||
# + TrigramSimilarity('description', search_text)
|
|
||||||
# + TrigramSimilarity('steps__ingredients__food__name', search_text)
|
|
||||||
# + TrigramSimilarity('keywords__name', search_text)
|
|
||||||
return (
|
return (
|
||||||
self.get_queryset()
|
self.get_queryset()
|
||||||
.annotate(
|
.annotate(
|
||||||
search=search_vectors,
|
search=search_vectors,
|
||||||
rank=search_rank,
|
rank=search_rank,
|
||||||
# trigram=trigram_name+trigram_description+trigram_food+trigram_keyword
|
|
||||||
# trigram_name=trigram_name,
|
|
||||||
# trigram_description=trigram_description,
|
|
||||||
# trigram_food=trigram_food,
|
|
||||||
# trigram_keyword=trigram_keyword
|
|
||||||
)
|
)
|
||||||
.filter(
|
.filter(
|
||||||
Q(search=search_query)
|
Q(search=search_query)
|
||||||
# | Q(trigram_name__gt=0.1)
|
|
||||||
# | Q(name__icontains=search_text)
|
|
||||||
# | Q(trigram_name__gt=0.2)
|
|
||||||
# | Q(trigram_description__gt=0.2)
|
|
||||||
# | Q(trigram_food__gt=0.2)
|
|
||||||
# | Q(trigram_keyword__gt=0.2)
|
|
||||||
)
|
)
|
||||||
.order_by('-rank'))
|
.order_by('-rank'))
|
||||||
|
|||||||
@@ -0,0 +1,163 @@
|
|||||||
|
# Generated by Django 4.1.9 on 2023-05-25 13:05
|
||||||
|
|
||||||
|
import cookbook.models
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.db.models.deletion
|
||||||
|
import django_prometheus.models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
('cookbook', '0188_space_no_sharing_limit'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Property',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('property_amount', models.DecimalField(decimal_places=4, default=0, max_digits=32)),
|
||||||
|
],
|
||||||
|
bases=(models.Model, cookbook.models.PermissionModelMixin),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='PropertyType',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('name', models.CharField(max_length=128)),
|
||||||
|
('unit', models.CharField(blank=True, max_length=64, null=True)),
|
||||||
|
('icon', models.CharField(blank=True, max_length=16, null=True)),
|
||||||
|
('description', models.CharField(blank=True, max_length=512, null=True)),
|
||||||
|
('category', models.CharField(blank=True, choices=[('NUTRITION', 'Nutrition'), ('ALLERGEN', 'Allergen'), ('PRICE', 'Price'), ('GOAL', 'Goal'), ('OTHER', 'Other')], max_length=64, null=True)),
|
||||||
|
('open_data_slug', models.CharField(blank=True, default=None, max_length=128, null=True)),
|
||||||
|
],
|
||||||
|
bases=(models.Model, cookbook.models.PermissionModelMixin),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='UnitConversion',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('base_amount', models.DecimalField(decimal_places=16, default=0, max_digits=32)),
|
||||||
|
('converted_amount', models.DecimalField(decimal_places=16, default=0, max_digits=32)),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('open_data_slug', models.CharField(blank=True, default=None, max_length=128, null=True)),
|
||||||
|
],
|
||||||
|
bases=(django_prometheus.models.ExportModelOperationsMixin('unit_conversion'), models.Model, cookbook.models.PermissionModelMixin),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='fdc_id',
|
||||||
|
field=models.CharField(blank=True, default=None, max_length=128, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='open_data_slug',
|
||||||
|
field=models.CharField(blank=True, default=None, max_length=128, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='preferred_shopping_unit',
|
||||||
|
field=models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='preferred_shopping_unit', to='cookbook.unit'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='preferred_unit',
|
||||||
|
field=models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='preferred_unit', to='cookbook.unit'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='properties_food_amount',
|
||||||
|
field=models.IntegerField(blank=True, default=100),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='properties_food_unit',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, to='cookbook.unit'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='supermarket',
|
||||||
|
name='open_data_slug',
|
||||||
|
field=models.CharField(blank=True, default=None, max_length=128, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='supermarketcategory',
|
||||||
|
name='open_data_slug',
|
||||||
|
field=models.CharField(blank=True, default=None, max_length=128, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unit',
|
||||||
|
name='base_unit',
|
||||||
|
field=models.TextField(blank=True, default=None, max_length=256, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unit',
|
||||||
|
name='open_data_slug',
|
||||||
|
field=models.CharField(blank=True, default=None, max_length=128, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='supermarketcategoryrelation',
|
||||||
|
constraint=models.UniqueConstraint(fields=('supermarket', 'category'), name='unique_sm_category_relation'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unitconversion',
|
||||||
|
name='base_unit',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='unit_conversion_base_relation', to='cookbook.unit'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unitconversion',
|
||||||
|
name='converted_unit',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='unit_conversion_converted_relation', to='cookbook.unit'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unitconversion',
|
||||||
|
name='created_by',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to=settings.AUTH_USER_MODEL),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unitconversion',
|
||||||
|
name='food',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to='cookbook.food'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='unitconversion',
|
||||||
|
name='space',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.space'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='propertytype',
|
||||||
|
name='space',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.space'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='property',
|
||||||
|
name='property_type',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='cookbook.propertytype'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='property',
|
||||||
|
name='space',
|
||||||
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.space'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='food',
|
||||||
|
name='properties',
|
||||||
|
field=models.ManyToManyField(blank=True, to='cookbook.property'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='recipe',
|
||||||
|
name='properties',
|
||||||
|
field=models.ManyToManyField(blank=True, to='cookbook.property'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='unitconversion',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'base_unit', 'converted_unit', 'food'), name='f_unique_conversion_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='propertytype',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'name'), name='property_type_unique_name_per_space'),
|
||||||
|
),
|
||||||
|
]
|
||||||
38
cookbook/migrations/0190_auto_20230525_1506.py
Normal file
38
cookbook/migrations/0190_auto_20230525_1506.py
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
# Generated by Django 4.1.9 on 2023-05-25 13:06
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
from django_scopes import scopes_disabled
|
||||||
|
from gettext import gettext as _
|
||||||
|
|
||||||
|
def migrate_old_nutrition_data(apps, schema_editor):
|
||||||
|
print('Transforming nutrition information, this might take a while on large databases')
|
||||||
|
with scopes_disabled():
|
||||||
|
PropertyType = apps.get_model('cookbook', 'PropertyType')
|
||||||
|
RecipeProperty = apps.get_model('cookbook', 'Property')
|
||||||
|
Recipe = apps.get_model('cookbook', 'Recipe')
|
||||||
|
Space = apps.get_model('cookbook', 'Space')
|
||||||
|
|
||||||
|
# TODO respect space
|
||||||
|
for s in Space.objects.all():
|
||||||
|
property_fat = PropertyType.objects.get_or_create(name=_('Fat'), unit=_('g'), space=s, )[0]
|
||||||
|
property_carbohydrates = PropertyType.objects.get_or_create(name=_('Carbohydrates'), unit=_('g'), space=s, )[0]
|
||||||
|
property_proteins = PropertyType.objects.get_or_create(name=_('Proteins'), unit=_('g'), space=s, )[0]
|
||||||
|
property_calories = PropertyType.objects.get_or_create(name=_('Calories'), unit=_('kcal'), space=s, )[0]
|
||||||
|
|
||||||
|
for r in Recipe.objects.filter(nutrition__isnull=False, space=s).all():
|
||||||
|
rp_fat = RecipeProperty.objects.create(property_type=property_fat, property_amount=r.nutrition.fats, space=s)
|
||||||
|
rp_carbohydrates = RecipeProperty.objects.create(property_type=property_carbohydrates, property_amount=r.nutrition.carbohydrates, space=s)
|
||||||
|
rp_proteins = RecipeProperty.objects.create(property_type=property_proteins, property_amount=r.nutrition.proteins, space=s)
|
||||||
|
rp_calories = RecipeProperty.objects.create(property_type=property_calories, property_amount=r.nutrition.calories, space=s)
|
||||||
|
r.properties.add(rp_fat, rp_carbohydrates, rp_proteins, rp_calories)
|
||||||
|
r.nutrition = None
|
||||||
|
r.save()
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('cookbook', '0189_property_propertytype_unitconversion_food_fdc_id_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RunPython(migrate_old_nutrition_data)
|
||||||
|
]
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
# Generated by Django 4.1.9 on 2023-06-20 13:07
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.db.models.deletion
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
('cookbook', '0190_auto_20230525_1506'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.SeparateDatabaseAndState(
|
||||||
|
database_operations=[
|
||||||
|
migrations.RunSQL(
|
||||||
|
sql="ALTER TABLE cookbook_food_properties RENAME TO cookbook_foodproperty",
|
||||||
|
reverse_sql="ALTER TABLE cookbook_foodproperty RENAME TO cookbook_food_properties",
|
||||||
|
),
|
||||||
|
],
|
||||||
|
state_operations=[
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='FoodProperty',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('food', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.food')),
|
||||||
|
('property', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.property')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='food',
|
||||||
|
name='properties',
|
||||||
|
field=models.ManyToManyField(blank=True, through='cookbook.FoodProperty', to='cookbook.property'),
|
||||||
|
),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='foodproperty',
|
||||||
|
constraint=models.UniqueConstraint(fields=('food', 'property'), name='property_unique_food'),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='property',
|
||||||
|
name='import_food_id',
|
||||||
|
field=models.IntegerField(blank=True, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='property',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'property_type', 'import_food_id'), name='property_unique_import_food_per_space'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
# Generated by Django 4.1.9 on 2023-06-20 13:30
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('cookbook', '0191_foodproperty_property_import_food_id_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='food',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='food_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='propertytype',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='property_type_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='supermarket',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='supermarket_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='supermarketcategory',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='supermarket_category_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='unit',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='unit_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
migrations.AddConstraint(
|
||||||
|
model_name='unitconversion',
|
||||||
|
constraint=models.UniqueConstraint(fields=('space', 'open_data_slug'), name='unit_conversion_unique_open_data_slug_per_space'),
|
||||||
|
),
|
||||||
|
]
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user