Generating a recipe graph with ChatGPT

Recipes have been structured in a similar format for a long time; a list of ingredients followed by the steps necessary to create a dish. While this works for human readers, it’s less convenient for computers. They struggle to understand when each ingredient is used and what happens in each step. This makes it challenging to design applications that provide step-by-step recipe instructions, let alone enhance the overall cooking process.

To combat this, a new format, Cooklang, was created, in which the text is organized in a straightforward structure and users could tag ingredients and tools as they were mentioned. This allowed the development of applications that could present recipes step by step, create shopping lists, and offer more benefits.

We attempted to create a universal tool that could import any recipe from the web and reformat it into this user-friendly layout. However, it was a difficult task due to the variety of ways people write recipes. For instance, a recipe might introduce a bunch of spices early on and later merely refer to them as “spices”, complicating the process of tracking individual ingredients.

Consequently, we opted for a different approach. We proposed utilizing AI/ML assistance to trace the use of ingredients from start to finish. This could provide a clear understanding of when and how each component is used, ultimately leading to the construction of a recipe graph. This graph begins with the ingredients and cookware, following their interactions through the recipe, visualizing the process as it evolves.

Not long ago, people approached me to present their recipes in a more structured format and engage in discussions about them. The graph format is particularly useful for computers. Check out this informative Twitter thread about different recipe representations:

Thus, I wanted to explore the possibility of using GPT-4 to create this graph.

GPT-4 works by responding to user prompts. For instance, you provide an example of the desired output, and it generates similar results. We decided to use this model for entity recognition and to identify relationships between ingredients and cookware.

We examined other projects for inspiration, like GraphGPT, which utilizes direct instructions to prompt the model. Adapting this approach, we had our initial attempt at creating a graph. However, it was less successful in recognizing relationships.

After a few tweaks, we managed to achieve better results. Here’s our improved attempt:

Me

You're a cook detective. Given a prompt, you will help me to list of all ingredient transformation in a recipe from start to finish and provide list of these transformations.

Present transformation in this format. First goes list of all input items required for the transformation: it can be ingredients or cookware. Then follows a transformation name, usually a verb. And at the end assign a new name to transformed ingredients, so you can reference it in next transformations as input ingredient. Output transformation in this format: [INGREDIENT 1, INGREDIENT 2, COOKWARE 1] (TRANSFORMATION NAME) => TRANSFORMED INGREDIENT NAME.

Example: Promt:

Ingredients
3 large free-range eggs
125 g plain flour
250 ml milk
unsalted butter

Method
Crack the eggs into a blender, then add the flour, milk and 1 pinch of sea salt, and blitz until smooth.
Pour into a bowl and leave to stand for 15 minutes.
Melt the butter in a large non-stick frying pan on a medium heat, then tilt the pan so the butter coats the surface.
Pour in 1 ladle of batter and tilt again, so that the batter spreads all over the base, then cook for 1 to 2 minutes, or until it starts to come away from the sides.
Once golden underneath, flip the pancake over and cook for 1 further minute, or until cooked through.
Serve straightaway with your favourite topping.

Transformations:

[
	["eggs", "blender"] ("crack into") => "blender with eggs",
	["blender with eggs", "flour", "milk", "sea salt"] ("blitz until smooth") => "pancakes batter in blender",
	["pancakes batter in blender", "bowl"] ("pour into") => "pancakes batter in bowl",
	["pancakes batter in bowl"] ("leave to stand for 15 minutes") => "rested pancakes batter",
	["butter", "large non-stick frying pan"] ("melt on a medium heat") => "pan with melted butter",
	["pan with melted butter"] ("tilt") => "pan with melted butter coated all surface",
	["rested pancakes batter", "pan with melted butter coated all surface"] ("pour in") => "batter in pan",
	["batter in pan"] ("tilt") => "batter spreaded in pan",
	["batter spreaded in pan"] ("cook until golden") => "one side of pancake is done",
	["one side of pancake is done"] ("flip") => "other side of pancake is cooking",
	["other side of pancake is cooking"] ("cook until golden") => "both side of pancake is done",
	["both side of pancake is done"] ("serve") => "pancake is done",
]

Promt: Cook your spaghetti in a large pan of salted boiling water according to the packet instructions. Meanwhile, peel and finely chop the garlic. Heat 3 good lugs of extra virgin olive oil in a large frying pan, then toss in the garlic and crumble in the chilli. As the garlic begins to colour, add the prawns and sauté them for a minute. Add the white wine and tomato purée, and simmer for a couple of minutes. Grate the zest of the lemon and roughly chop the rocket. When the pasta is ready, drain it in a colander, reserving a little of the cooking water. Toss the spaghetti with the sauce, squeeze in the lemon juice, add half the chopped rocket, adding a little of the reserved cooking water if you want to loosen the sauce a bit, and correct the seasoning with sea salt and black pepper. Divide between 4 plates and sprinkle with the grated lemon zest and the rest of the rocket leaves. Transformations:

ChatGPT

[
  ["spaghetti", "large pan", "salted boiling water"] ("cook according to packet instructions") => "cooked spaghetti",
  ["garlic"] ("peel and finely chop") => "finely chopped garlic",
  ["extra virgin olive oil", "large frying pan", "garlic", "chilli"] ("heat in") => "pan with heated olive oil, garlic, and chilli",
  ["pan with heated olive oil, garlic, and chilli", "prawns"] ("add") => "pan with heated olive oil, garlic, chilli, and prawns",
  ["white wine", "tomato purée"] ("add") => "pan with heated olive oil, garlic, chilli, prawns, white wine, and tomato purée",
  ["lemon"] ("grate zest of") => "grated lemon zest",
  ["rocket"] ("roughly chop") => "chopped rocket",
  ["cooked spaghetti", "colander"] ("drain in") => "drained spaghetti",
  ["drained spaghetti", "sauce", "lemon juice", "chopped rocket"] ("toss with") => "spaghetti with sauce, lemon juice, and chopped rocket",
  ["spaghetti with sauce, lemon juice, and chopped rocket", "reserved cooking water"] ("add if desired to loosen sauce") => "spaghetti with sauce, lemon juice, chopped rocket, and reserved cooking water",
  ["spaghetti with sauce, lemon juice, chopped rocket, and reserved cooking water"] ("correct seasoning with sea salt and black pepper") => "seasoned spaghetti with sauce, lemon juice, chopped rocket, and reserved cooking water",
  ["seasoned spaghetti with sauce, lemon juice, chopped rocket, and reserved cooking water"] ("divide between 4 plates") => "plated spaghetti with sauce, lemon juice, chopped rocket, and reserved cooking water",
  ["grated lemon zest", "rest of rocket leaves"] ("sprinkle with") => "sprinkled spaghetti with sauce, lemon zest, chopped rocket, and reserved cooking water"
]

Further fine-tuning allowed us to generate Ruby-compatible syntax for creating a graph image. Here’s the result:

Me

You're a cook detective. Given a prompt, you will help me to list of all ingredient transformation in a recipe from start to finish and provide list of these transformations.

Present transformation in this format. First goes list of all input items required for the transformation: it can be ingredients or cookware. Then follows a transformation name, usually a verb. And at the end assign a new name to transformed ingredients, so you can reference it in next transformations as input ingredient. Output transformation in this format: { input: ["INGREDIENT 1", "INGREDIENT 2", "COOKWARE 1"], transformation: "TRANSFORMATION NAME", output: ["TRANSFORMED INGREDIENT NAME 1", "TRANSFORMED INGREDIENT NAME 2"] }.

Example: Promt:

Ingredients
3 large free-range eggs
125 g plain flour
250 ml milk
unsalted butter

Method
Crack the eggs into a blender, then add the flour, milk and 1 pinch of sea salt, and blitz until smooth.
Pour into a bowl and leave to stand for 15 minutes.
Melt the butter in a large non-stick frying pan on a medium heat, then tilt the pan so the butter coats the surface.
Pour in 1 ladle of batter and tilt again, so that the batter spreads all over the base, then cook for 1 to 2 minutes, or until it starts to come away from the sides.
Once golden underneath, flip the pancake over and cook for 1 further minute, or until cooked through.
Serve straightaway with your favourite topping.

Transformations:

[
  { input: ["eggs", "blender"], transformation: "crack into", output: "blender with eggs" },
  { input: ["blender with eggs", "flour", "milk", "sea salt"], transformation: "blitz until smooth", output: "pancakes batter in blender" },
  { input: ["pancakes batter in blender", "bowl"], transformation: "pour into", output: "pancakes batter in bowl" },
  { input: ["pancakes batter in bowl"], transformation: "leave to stand for 15 minutes", output: "rested pancakes batter" },
  { input: ["butter", "large non-stick frying pan"], transformation: "melt on a medium heat", output: "pan with melted butter" },
  { input: ["pan with melted butter"], transformation: "tilt", output: "pan with melted butter coated all surface" },
  { input: ["rested pancakes batter", "pan with melted butter coated all surface"], transformation: "pour in", output: "batter in pan" },
  { input: ["batter in pan"], transformation: "tilt", output: "batter spreaded in pan" },
  { input: ["batter spreaded in pan"], transformation: "cook until golden", output: "one side of pancake is done" },
  { input: ["one side of pancake is done"], transformation: "flip", output: "other side of pancake is cooking" },
  { input: ["other side of pancake is cooking"], transformation: "cook until golden", output: "both side of pancake is done" },
  { input: ["both side of pancake is done"], transformation: "serve", output: "pancake is done" },
]

Promt:

Ingredients
½ x 7 g sachet of yeast
125 g rice flour
olive oil
1 heaped teaspoon jalfrezi curry paste
½–1 fresh red chilli
1 bunch of spring onions
½ x 660 g jar of chickpeas
100 g baby spinach
4 cm piece of ginger
2 ripe tomatoes
1 lime
2 tablespoons unsweetened desiccated coconut flakes
½ a bunch of fresh coriander , (15g)
4 large free-range eggs
4 heaped tablespoons natural yoghurt

Method
In a bowl, whisk ½ a 7g sachet of yeast yeast into 250ml of lukewarm water, then gradually whisk in 125g of rice flour and a pinch of sea salt until smooth. Put aside.
Place a frying pan on a medium heat with 1 teaspoon of oil and 1 heaped teaspoon of jalfrezi curry paste.
Finely slice half the fresh red chilli (whether you're using ½ or 1 chilli, to your taste) and just the top green halves of 1 bunch of spring onions and add to the pan. Cook and stir for 5 minutes, then add ½ x 660g jar of chickpeas (juice and all), and 100g of baby spinach. Bring to the boil, then simmer on a low heat until needed.
To make the salsa, trim and finely slice the whites of the spring onions, along with the remaining chilli and place in a bowl. Peel and finely grate in a 4cm piece of ginger. Quarter and deseed 2 ripe tomatoes, then finely dice and add the flesh to the bowl. Finely grate over the zest and squeeze in the juice from 1 lime, add 2 tablespoons of unsweetened desiccated coconut flakes and the leaves from ½ a bunch of fresh coriander (15g). Mix well, then taste and season to perfection.
For the pancakes, place a small non-stick frying pan on a medium heat, drizzle with a little oil, then carefully wipe it around and out with a ball of kitchen paper.
Stirring your batter well each time, add a quarter to the pan, swiftly swirling it around and up the sides to create extra crispy bits. Cook for about 5 minutes on the underside only, until dark golden – have faith, and when it’s coloured, use a spatula to ease it away from the sides and slide it out. Repeat with the remaining batter, and just before you start the last pancake, poach 4 eggs to your liking in a large pan of simmering water.
Serve each pancake with a spoonful of curried chickpeas and coconut salsa, 1 heaped tablespoon of natural yoghurt, and with a poached egg sitting proudly on top.

Transformations:

ChatGPT

Here are the transformations for the given prompt:

[
  { input: ["½ x 7 g sachet of yeast", "bowl", "lukewarm water"], transformation: "whisk into", output: "yeast water" },
  { input: ["yeast water", "125 g rice flour", "pinch of sea salt"], transformation: "gradually whisk in", output: "smooth batter" },
  { input: ["smooth batter"], transformation: "put aside", output: "rested batter" },
  { input: ["frying pan", "1 teaspoon of oil", "1 heaped teaspoon of jalfrezi curry paste"], transformation: "heat on medium", output: "pan with curry paste" },
  { input: ["½–1 fresh red chilli", "1 bunch of spring onions"], transformation: "finely slice", output: ["sliced chilli", "sliced spring onions"] },
  { input: ["pan with curry paste", "sliced chilli", "sliced spring onions"], transformation: "add and cook for 5 minutes", output: "cooked vegetables in pan" },
  { input: ["cooked vegetables in pan", "½ x 660 g jar of chickpeas", "100 g baby spinach"], transformation: "add and bring to boil", output: "boiling chickpeas and spinach" },
  { input: ["boiling chickpeas and spinach"], transformation: "simmer on low heat", output: "simmered chickpeas and spinach" },
  { input: ["spring onions", "remaining chilli", "4cm piece of ginger", "2 ripe tomatoes", "1 lime", "2 tablespoons unsweetened desiccated coconut flakes", "½ a bunch of fresh coriander"], transformation: "prepare salsa", output: "salsa" },
  { input: ["small non-stick frying pan", "oil"], transformation: "drizzle and wipe with kitchen paper", output: "prepared pan" },
  { input: ["rested batter", "prepared pan"], transformation: "cook pancakes", output: "pancakes" },
  { input: ["4 large free-range eggs"], transformation: "poach", output: "poached eggs" },
  { input: ["pancakes", "simmered chickpeas and spinach", "salsa", "1 heaped tablespoon of natural yoghurt", "poached eggs"], transformation: "assemble", output: "finished meal" }
]

Notice how the AI smartly creates names for intermediate ingredients.

Open full graph

What’s next? If this system works effectively, it could revolutionize the way we approach recipe management. It isn’t about creating a new language syntax, but rather focusing on improving tools and user experiences. This graph-based approach could easily be integrated into our CLI and mobile applications. This means users won’t need to learn any syntax. They can continue writing recipes as they always have, and the computer will extract what it needs, making everyone’s lives easier.

-Alexey