added mealie importer
This commit is contained in:
parent
0f251bee9b
commit
5d1d6d4248
@ -5,6 +5,7 @@
|
|||||||
<w>csrftoken</w>
|
<w>csrftoken</w>
|
||||||
<w>gunicorn</w>
|
<w>gunicorn</w>
|
||||||
<w>ical</w>
|
<w>ical</w>
|
||||||
|
<w>mealie</w>
|
||||||
<w>traefik</w>
|
<w>traefik</w>
|
||||||
</words>
|
</words>
|
||||||
</dictionary>
|
</dictionary>
|
||||||
|
@ -135,8 +135,9 @@ class ImportExportBase(forms.Form):
|
|||||||
DEFAULT = 'DEFAULT'
|
DEFAULT = 'DEFAULT'
|
||||||
PAPRIKA = 'PAPRIKA'
|
PAPRIKA = 'PAPRIKA'
|
||||||
NEXTCLOUD = 'NEXTCLOUD'
|
NEXTCLOUD = 'NEXTCLOUD'
|
||||||
|
MEALIE = 'MEALIE'
|
||||||
|
|
||||||
type = forms.ChoiceField(choices=((DEFAULT, _('Default')), (PAPRIKA, _('Paprika')), (NEXTCLOUD, _('Nextcloud Cookbook')),))
|
type = forms.ChoiceField(choices=((DEFAULT, _('Default')), (PAPRIKA, _('Paprika')), (NEXTCLOUD, _('Nextcloud Cookbook')), (MEALIE, _('Mealie')),))
|
||||||
|
|
||||||
|
|
||||||
class ImportForm(ImportExportBase):
|
class ImportForm(ImportExportBase):
|
||||||
|
56
cookbook/integration/mealie.py
Normal file
56
cookbook/integration/mealie.py
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import json
|
||||||
|
import re
|
||||||
|
from io import BytesIO
|
||||||
|
from zipfile import ZipFile
|
||||||
|
|
||||||
|
from rest_framework.renderers import JSONRenderer
|
||||||
|
|
||||||
|
from cookbook.helper.ingredient_parser import parse
|
||||||
|
from cookbook.integration.integration import Integration
|
||||||
|
from cookbook.models import Recipe, Step, Food, Unit, Ingredient
|
||||||
|
from cookbook.serializer import RecipeExportSerializer
|
||||||
|
|
||||||
|
|
||||||
|
class Mealie(Integration):
|
||||||
|
|
||||||
|
def import_file_name_filter(self, zip_info_object):
|
||||||
|
print("testing", zip_info_object.filename)
|
||||||
|
return re.match(r'^recipes/([A-Za-z\d-])+.json$', zip_info_object.filename)
|
||||||
|
|
||||||
|
def get_recipe_from_file(self, file):
|
||||||
|
recipe_json = json.loads(file.getvalue().decode("utf-8"))
|
||||||
|
|
||||||
|
recipe = Recipe.objects.create(
|
||||||
|
name=recipe_json['name'].strip(), description=recipe_json['description'].strip(),
|
||||||
|
created_by=self.request.user, internal=True)
|
||||||
|
|
||||||
|
# TODO parse times (given in PT2H3M )
|
||||||
|
|
||||||
|
ingredients_added = False
|
||||||
|
for s in recipe_json['recipeInstructions']:
|
||||||
|
step = Step.objects.create(
|
||||||
|
instruction=s['text']
|
||||||
|
)
|
||||||
|
if not ingredients_added:
|
||||||
|
ingredients_added = True
|
||||||
|
|
||||||
|
for ingredient in recipe_json['recipeIngredient']:
|
||||||
|
amount, unit, ingredient, note = parse(ingredient)
|
||||||
|
f, created = Food.objects.get_or_create(name=ingredient)
|
||||||
|
u, created = Unit.objects.get_or_create(name=unit)
|
||||||
|
step.ingredients.add(Ingredient.objects.create(
|
||||||
|
food=f, unit=u, amount=amount, note=note
|
||||||
|
))
|
||||||
|
recipe.steps.add(step)
|
||||||
|
|
||||||
|
for f in self.files:
|
||||||
|
if '.zip' in f.name:
|
||||||
|
import_zip = ZipFile(f.file)
|
||||||
|
for z in import_zip.filelist:
|
||||||
|
if re.match(f'^images/{recipe_json["slug"]}.jpg$', z.filename):
|
||||||
|
self.import_recipe_image(recipe, BytesIO(import_zip.read(z.filename)))
|
||||||
|
|
||||||
|
return recipe
|
||||||
|
|
||||||
|
def get_file_from_recipe(self, recipe):
|
||||||
|
raise NotImplementedError('Method not implemented in storage integration')
|
@ -54,6 +54,4 @@ class NextcloudCookbook(Integration):
|
|||||||
return recipe
|
return recipe
|
||||||
|
|
||||||
def get_file_from_recipe(self, recipe):
|
def get_file_from_recipe(self, recipe):
|
||||||
export = RecipeExportSerializer(recipe).data
|
raise NotImplementedError('Method not implemented in storage integration')
|
||||||
|
|
||||||
return 'recipe.json', JSONRenderer().render(export).decode("utf-8")
|
|
||||||
|
@ -7,6 +7,7 @@ from django.utils.translation import gettext as _
|
|||||||
from cookbook.forms import ExportForm, ImportForm, ImportExportBase
|
from cookbook.forms import ExportForm, ImportForm, ImportExportBase
|
||||||
from cookbook.helper.permission_helper import group_required
|
from cookbook.helper.permission_helper import group_required
|
||||||
from cookbook.integration.default import Default
|
from cookbook.integration.default import Default
|
||||||
|
from cookbook.integration.mealie import Mealie
|
||||||
from cookbook.integration.nextcloud_cookbook import NextcloudCookbook
|
from cookbook.integration.nextcloud_cookbook import NextcloudCookbook
|
||||||
from cookbook.integration.paprika import Paprika
|
from cookbook.integration.paprika import Paprika
|
||||||
from cookbook.models import Recipe
|
from cookbook.models import Recipe
|
||||||
@ -19,6 +20,8 @@ def get_integration(request, export_type):
|
|||||||
return Paprika(request)
|
return Paprika(request)
|
||||||
if export_type == ImportExportBase.NEXTCLOUD:
|
if export_type == ImportExportBase.NEXTCLOUD:
|
||||||
return NextcloudCookbook(request)
|
return NextcloudCookbook(request)
|
||||||
|
if export_type == ImportExportBase.MEALIE:
|
||||||
|
return Mealie(request)
|
||||||
|
|
||||||
|
|
||||||
@group_required('user')
|
@group_required('user')
|
||||||
|
@ -48,6 +48,14 @@ You will get a `Recipes.zip` file. Simply upload the file and choose the Nextclo
|
|||||||
└── full.jpg
|
└── full.jpg
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Mealie
|
||||||
|
Mealie provides structured data similar to nextcloud.
|
||||||
|
|
||||||
|
To migrate your recipes
|
||||||
|
|
||||||
|
1. Go to you Mealie settings and create a new Backup
|
||||||
|
2. Download the backup by clicking on it and pressing download (this wasn't working for me, so I had to manually pull it from the server)
|
||||||
|
3. Upload the entire `.zip` file to the importer page and import everything
|
||||||
|
|
||||||
## Paprika
|
## Paprika
|
||||||
Paprika can create two types of export. The first is a proprietary `.paprikarecipes` file in some kind of binarized format.
|
Paprika can create two types of export. The first is a proprietary `.paprikarecipes` file in some kind of binarized format.
|
||||||
|
Loading…
Reference in New Issue
Block a user