eliasbrange.dev
Automatically generate grocery list with Trello and AWS Lambda

Automatically generate grocery list with Trello and AWS Lambda

I use Trello for all kinds of personal stuff, one of which is keeping track of recipes and grocery lists. After planning which meals to cook and buy groceries for, a lot of manual work goes into checking every recipe and adding the ingredients to the grocery shopping list. Manual stuff is tedious, so I automated it with AWS Lambda. Read on to find out how!

The Trello board

Info

Our Trello board is in Swedish, so the screenshots will be in Swedish as well.

We have a Trello board that contains a lot of recipes grouped into lists, such as Pasta Dishes, Soups & Stews, and so on. We also have a list of groceries that needs to be bought, so that we always have the shopping list available when shopping.

Trello board with recipe lists for different types of dishes
Trello board with recipe lists for different types of dishes

In the above picture, you can get a glimpse of the Trello board, with each card containing a picture of the dish as well as a link to the external site where the recipe can be found.

What I wanted to automate was the following:

Below you can see the completed automation in action:

Automatically generated grocery list
Automatically generated grocery list

Trello setup: Part 1

I started by adding two new lists to my board, one to hold recipes that should be bought, and one for the generated grocery shopping list.

I then proceeded to add a checklist to recipe cards. This checklist contains the ingredients required for a single serving of the recipe. For example, if a recipe requires one onion for four servings, I would add onion:0.25. Here I had to be consistent between recipes so that the units used for ingredients would be the same across all recipes.

The unit for onion here is number of onions, while for pasta I use gram and for cooking cream I use deciliter.

For ingredients where I do not care about the unit and just need to make sure that I buy it if needed, such as cooking oil and certain spices, I set the amount needed to 0.

With the checklist(s) completed, I went on and created a Card button in the Automation menu. When clicked, this button triggers an action that adds a new card to the recipes to buy list. The created card gets the following title ANTAL_PORTIONER:{cardidlong}:{cardname}. The ANTAL_PORTIONER (number of servings in Swedish) is a placeholder that needs to be changed to an actual number before generating the shopping list.

The cards now look like this:

A recipe card with an ingredient list and custom automation button
A recipe card with an ingredient list and custom automation button

Clicking the button creates a new card in the recipes to buy list, and I then edit the text ANTAL_PORTIONER in the title to the number of servings I need to purchase.

Recipes added to the "Recipes to buy" list with a placeholder number of servings
Recipes added to the "Recipes to buy" list with a placeholder number of servings
Recipes added to the "Recipes to buy" list with a specified number of servings
Recipes added to the "Recipes to buy" list with a specified number of servings

I also needed to get an API key and token for the Trello API, as described here.

I also needed to fetch the IDs of both the recipes to buy list, the grocery shopping list, and the board itself. This was easiest done by creating a new card in each of the lists, clicking on them and then adding .json to the end of the URL in the browser. From there I could then extract the idList and idBoard variables.

AWS part

This whole thing really started with me wanting to find a use case for the newly released function URLs of AWS Lambda. So I decided to go with FastAPI in a Lambda function (as I usually do).

Since the API should be reachable from Trello I decided to keep the function URL open to the public. To secure it, I added code to the Lambda function that validates the X-Api-Key header in incoming requests.

Secrets

To not require hard-coded secrets in the Lambda function, I added the following parameters to the AWS Systems Manager Parameter Store:

Lambda code

Folder structure

The code examples below are structured as follows:

1
TrelloMatApi/
2
├── template.yml
3
├── api-function
4
│ ├── requirements.txt
5
│ ├── src
6
│ │ ├── __init__.py
7
│ │ ├── models.py
8
│ │ ├── trello.py
9
│ │ ├── utils.py

requirements.txt

The Lambda function requires the following libraries.

1
mangum
2
fastapi
3
pydantic
4
py-trello
5
aws-lambda-powertools

__init__.py

The handler function is a FastAPI application wrapped by a Mangum adapter. The API includes a single route POST /generate, that neither requires nor returns any payload.

Authorization is done with the X-Api-Key header, which is validated by the auth method. The valid key is fetched from AWS Systems Manager Parameter Store and cached between invocations.

1
from fastapi import FastAPI, HTTPException, Header, Depends
2
from mangum import Mangum
3
from aws_lambda_powertools.utilities import parameters
4
from src.utils import logger, tracer
5
from src import trello
6
7
app = FastAPI()
8
API_KEY = parameters.get_parameter("/trellomat/aws_api_key")
9
10
11
async def auth(x_api_key: str = Header(...)):
12
if x_api_key != API_KEY:
13
raise HTTPException(status_code=401, detail="Unauthorized")
14
15
16
@app.post("/generate", status_code=204, dependencies=[Depends(auth)])
17
def generate():
18
trello.generate()
19
20
21
handler = Mangum(app)
22
handler.__name__ = "handler"
23
handler = tracer.capture_lambda_handler(handler)
24
handler = logger.inject_lambda_context(handler, clear_state=True)

utils.py

This is a simple file that sets up a logger and tracer using AWS Lambda Power Tools.

1
from aws_lambda_powertools import Logger, Tracer
2
3
4
logger: Logger = Logger()
5
tracer: Tracer = Tracer()

models.py

This is where I define my ingredients.

In the INGREDIENTS dict, I specify ingredients their units as well as which category the ingredient is part of.

The Category decides which label an ingredient gets, and it is also used for sorting the generated shopping list. The default values in the Ingredient class are there to not break the API whenever an ingredient is added on the Trello side before adding it to the Lambda function itself.

1
from enum import Enum
2
from pydantic import BaseModel
3
4
5
class Category(Enum):
6
GREEN = 0
7
FRIDGE = 1
8
BREAD = 2
9
FROZEN = 3
10
DRY = 4
11
OTHER = 5
12
NONE = 6
13
14
15
class Ingredient(BaseModel):
16
name: str
17
unit: str = "NOUNIT"
18
cat: Category = Category.NONE
19
amount: int = 0
20
21
22
INGREDIENTS = {
23
"vitlök": {
24
"unit": "klyftor",
25
"cat": Category.GREEN,
26
},
27
"gul lök": {
28
"unit": "st",
29
"cat": Category.GREEN,
30
},
31
"matlagningsgrädde": {
32
"unit": "dl",
33
"cat": Category.FRIDGE,
34
},
35
...
36
}

trello.py

This is where the magic happens. First, all required information is fetched from AWS Systems Manager Parameter Store.

We then initialize a TrelloClient from the py-trello library, as well as instantiate objects for the board and lists.

The generate function then:

  1. Archives all existing cards in the grocery shopping list.
  2. For each card in the recipes to buy list (which has the format {servings}:{recipe_card_id}:{recipe_name}), it fetches the card with the recipe_card_id.
  3. On this card, it looks for the checklist named Ingredienser.
  4. For each checklist item, it adds the required ingredient amount to the ingredients dict. It also adds the ingredient unit and category from the model defined earlier.
  5. The ingredients dict is then sorted by category.
  6. For each ingredient in the ingredients dict, it creates a new card in the grocery shopping list.
1
from aws_lambda_powertools.utilities import parameters
2
from trello import TrelloClient
3
from src.utils import logger
4
from src.models import INGREDIENTS, Ingredient, Category
5
6
7
TOKEN = parameters.get_parameter("/trellomat/token")
8
API_KEY = parameters.get_parameter("/trellomat/api_key")
9
BOARD_ID = parameters.get_parameter("/trellomat/board_id")
10
RECIPE_LIST_ID = parameters.get_parameter("/trellomat/recipe_list_id")
11
SHOPPING_LIST_ID = parameters.get_parameter("/trellomat/shopping_list_id")
12
CHECKLIST_KEY = "Ingredienser"
13
14
client = TrelloClient(api_key=API_KEY, token=TOKEN)
15
board = client.get_board(board_id=BOARD_ID)
16
recipe_list = client.get_list(list_id=RECIPE_LIST_ID)
17
shopping_list = client.get_list(list_id=SHOPPING_LIST_ID)
18
labels = board.get_labels()
19
20
21
def _get_label(cat: Category) -> list:
22
try:
23
return [labels[cat.value]]
24
except IndexError:
25
return []
26
27
28
def _archive_old_cards():
29
logger.info("Clearing AUTO shopping list")
30
shopping_list.archive_all_cards()
31
32
33
def _get_ingredient_list() -> list[Ingredient]:
34
logger.info("Getting ingredients")
35
36
cards = {card.id: card for card in board.get_cards()}
37
ingredients = {}
38
39
for card in recipe_list.list_cards_iter():
40
servings, ref_card_id, name = card.name.split(":")
41
ref_card = cards[ref_card_id]
42
43
logger.info(f"Reading recipe for {name}")
44
45
card_ingredients = next(
46
(cl for cl in ref_card.checklists if cl.name == CHECKLIST_KEY), None
47
)
48
49
if not card_ingredients:
50
logger.warning(f"No checklist named {CHECKLIST_KEY} found on card {name}")
51
continue
52
53
for item in card_ingredients.items:
54
name, amount = item["name"].split(":")
55
56
if name not in ingredients:
57
ingredients[name] = Ingredient(name=name, **INGREDIENTS.get(name, {}))
58
59
ingredients[name].amount += int(servings) * float(amount)
60
61
# Return list of Ingredients sorted by category
62
return sorted(ingredients.values(), key=lambda item: item.cat.value)
63
64
65
def _create_cards(ingredients: list[Ingredient]):
66
for ingredient in ingredients:
67
if ingredient.amount == 0:
68
title = f"{ingredient.name}"
69
elif ingredient.amount == int(ingredient.amount):
70
title = f"{int(ingredient.amount)} {ingredient.unit} {ingredient.name}"
71
else:
72
title = f"{ingredient.amount:.2f} {ingredient.unit} {ingredient.name}"
73
74
logger.info("Adding card: %s", title)
75
shopping_list.add_card(title, labels=_get_label(ingredient.cat))
76
77
78
def generate():
79
_archive_old_cards()
80
ingredients = _get_ingredient_list()
81
_create_cards(ingredients)
82
logger.info("Grocery list generated successfully")

template.yml

I deployed the Lambda function with AWS SAM. The SAM template looks like the following:

1
AWSTemplateFormatVersion: "2010-09-09"
2
Transform: AWS::Serverless-2016-10-31
3
Description: Trello Mat API
4
5
Resources:
6
ApiFunction:
7
Type: AWS::Serverless::Function
8
Properties:
9
MemorySize: 256
10
Timeout: 180
11
Tracing: Active
12
FunctionUrlConfig:
13
AuthType: NONE
14
CodeUri: api-function
15
Handler: src.handler
16
Runtime: python3.9
17
Policies:
18
- SSMParameterReadPolicy:
19
ParameterName: "trellomat/aws_api_key"
20
- SSMParameterReadPolicy:
21
ParameterName: "trellomat/token"
22
- SSMParameterReadPolicy:
23
ParameterName: "trellomat/api_key"
24
- SSMParameterReadPolicy:
25
ParameterName: "trellomat/board_id"
26
- SSMParameterReadPolicy:
27
ParameterName: "trellomat/recipe_list_id"
28
- SSMParameterReadPolicy:
29
ParameterName: "trellomat/shopping_list_id"
30
Environment:
31
Variables:
32
POWERTOOLS_SERVICE_NAME: TrelloMatAPI
33
34
Outputs:
35
FunctionUrl:
36
Description: URL of the Function
37
Value: !GetAtt ApiFunctionUrl.FunctionUrl

Trello setup: Part 2

With the API complete, I created a new Board button in the Automation menu. When clicked, a post request is sent to the Lambda function URL, with the API key specified in the X-Api-Key header. The Lambda function then reads the recipes to buy list and generates a grocery shopping list.

Board button to call AWS Lambda
Board button to call AWS Lambda

Final result

Lo and behold, the final result! After adding a few recipes to the recipes to buy list, I can now automatically generate a grocery shopping list.

Automatically generated grocery list
Automatically generated grocery list

About the author

I'm Elias Brange, a Cloud Consultant and AWS Community Builder in the Serverless category. I'm on a mission to drive Serverless adoption and help others on their Serverless AWS journey.

Did you find this article helpful? Share it with your friends and colleagues using the buttons below. It could help them too!

Are you looking for more content like this? Follow me on LinkedIn & Twitter !