1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
pub
// Re-export UniFFI types when feature is enabled
pub use *;
// Public API re-exports
pub use AiConfig;
pub use ;
pub use ImportError;
pub use ImageSource;
pub use RecipeComponents;
// Advanced builder API (for users who need more control)
pub use ;
/// Extract recipe components from a URL.
///
/// Returns `RecipeComponents` with text, metadata, and name fields.
/// Empty strings are used for fields that couldn't be extracted.
///
/// # Arguments
/// * `url` - The URL of the recipe webpage to fetch
///
/// # Returns
/// * `Ok(RecipeComponents)` - The extracted recipe components
/// * `Err(ImportError)` - If the URL cannot be fetched or parsed
///
/// # Example
/// ```no_run
/// use cooklang_import::url_to_recipe;
///
/// #[tokio::main]
/// async fn main() -> Result<(), Box<dyn std::error::Error>> {
/// let recipe = url_to_recipe("https://example.com/recipe").await?;
/// println!("Name: {}", recipe.name);
/// println!("Text: {}", recipe.text);
/// Ok(())
/// }
/// ```
pub async
/// Extract recipe components from images.
///
/// Returns `RecipeComponents` with text extracted via OCR.
/// Metadata contains the source info, name is typically empty.
///
/// Requires GOOGLE_API_KEY environment variable to be set.
///
/// # Arguments
/// * `images` - Vector of image sources (paths or base64-encoded data)
///
/// # Returns
/// * `Ok(RecipeComponents)` - The extracted recipe components
/// * `Err(ImportError)` - If OCR fails
///
/// # Example
/// ```no_run
/// use cooklang_import::{image_to_recipe, ImageSource};
///
/// #[tokio::main]
/// async fn main() -> Result<(), Box<dyn std::error::Error>> {
/// let images = vec![ImageSource::Path("/path/to/recipe.jpg".to_string())];
/// let recipe = image_to_recipe(&images).await?;
/// println!("Text: {}", recipe.text);
/// Ok(())
/// }
/// ```
pub async
/// Parse text into recipe components.
///
/// If `extract` is true, uses LLM to extract structured recipe data.
/// If `extract` is false, parses the text assuming it's already formatted
/// with optional YAML frontmatter.
///
/// # Arguments
/// * `text` - The recipe text
/// * `extract` - Whether to use LLM extraction
///
/// # Returns
/// * `Ok(RecipeComponents)` - The parsed recipe components
/// * `Err(ImportError)` - If parsing or extraction fails
///
/// # Example
/// ```no_run
/// use cooklang_import::text_to_recipe;
///
/// #[tokio::main]
/// async fn main() -> Result<(), Box<dyn std::error::Error>> {
/// let text = "2 eggs\n1 cup flour\n\nMix and bake at 350F.";
/// let recipe = text_to_recipe(text, false).await?;
/// println!("Text: {}", recipe.text);
/// Ok(())
/// }
/// ```
pub async
/// Convert recipe text to Cooklang format.
///
/// Takes recipe text (ingredients + instructions) and converts it
/// to Cooklang format using an LLM. Returns the Cooklang text
/// with optional YAML frontmatter if metadata/name are provided.
///
/// # Arguments
/// * `components` - The recipe components to convert
///
/// # Returns
/// * `Ok(String)` - The recipe in Cooklang format
/// * `Err(ImportError)` - If conversion fails
///
/// # Example
/// ```no_run
/// use cooklang_import::{text_to_cooklang, RecipeComponents};
///
/// #[tokio::main]
/// async fn main() -> Result<(), Box<dyn std::error::Error>> {
/// let components = RecipeComponents {
/// text: "2 eggs\n1 cup flour\n\nMix and bake at 350F.".to_string(),
/// metadata: String::new(),
/// name: "Simple Cake".to_string(),
/// };
/// let cooklang = text_to_cooklang(&components).await?;
/// println!("{}", cooklang);
/// Ok(())
/// }
/// ```
pub async