Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
aztec_process.cpp
Go to the documentation of this file.
1#ifndef __wasm__
2#include "aztec_process.hpp"
11#include <filesystem>
12#include <fstream>
13#include <iomanip>
14#include <nlohmann/json.hpp>
15#include <sstream>
16#include <thread>
17
18#ifdef ENABLE_AVM_TRANSPILER
19// Include avm_transpiler header
20#include <avm_transpiler.h>
21#endif
22
23namespace bb {
24
25namespace {
26
30std::vector<uint8_t> extract_bytecode(const nlohmann::json& function)
31{
32 if (!function.contains("bytecode")) {
33 throw_or_abort("Function missing bytecode field");
34 }
35
36 const auto& base64_bytecode = function["bytecode"].get<std::string>();
37 return decode_bytecode(base64_bytecode);
38}
39
43std::string compute_bytecode_hash(const std::vector<uint8_t>& bytecode)
44{
46 std::ostringstream oss;
47 for (auto byte : hash) {
48 oss << std::hex << std::setw(2) << std::setfill('0') << static_cast<int>(byte);
49 }
50 return oss.str();
51}
52
56std::filesystem::path get_cache_dir()
57{
58 const char* home = std::getenv("HOME");
59 if (!home) {
60 home = ".";
61 }
62 std::filesystem::path cache_dir = std::filesystem::path(home) / ".bb" / BB_VERSION / "vk_cache";
63 std::filesystem::create_directories(cache_dir);
64 return cache_dir;
65}
66
70bool is_private_constrained_function(const nlohmann::json& function)
71{
72 bool is_public = false;
73 bool is_unconstrained = false;
74
75 // Check custom_attributes for "public"
76 if (function.contains("custom_attributes") && function["custom_attributes"].is_array()) {
77 for (const auto& attr : function["custom_attributes"]) {
78 if (attr.is_string() && attr.get<std::string>() == "public") {
79 is_public = true;
80 break;
81 }
82 }
83 }
84
85 // Check is_unconstrained
86 if (function.contains("is_unconstrained") && function["is_unconstrained"].is_boolean()) {
87 is_unconstrained = function["is_unconstrained"].get<bool>();
88 }
89
90 return !is_public && !is_unconstrained;
91}
92
96std::vector<uint8_t> get_or_generate_cached_vk(const std::filesystem::path& cache_dir,
97 const std::string& circuit_name,
98 const std::vector<uint8_t>& bytecode,
99 bool force)
100{
101 std::string hash_str = compute_bytecode_hash(bytecode);
102 std::filesystem::path vk_cache_path = cache_dir / (hash_str + ".vk");
103
104 // Check cache unless force is true
105 if (!force && std::filesystem::exists(vk_cache_path)) {
106 info("Verification key already in cache: ", hash_str);
107 return read_file(vk_cache_path);
108 }
109
110 // Generate new VK
111 info("Generating verification key: ", hash_str);
112 auto response =
113 bbapi::ChonkComputeStandaloneVk{ .circuit = { .name = circuit_name, .bytecode = bytecode } }.execute();
114
115 // Cache the VK
116 write_file(vk_cache_path, response.bytes);
117
118 return response.bytes;
119}
120
124void generate_vks_for_functions(const std::filesystem::path& cache_dir,
126 bool force)
127{
128#ifdef __wasm__
129 throw_or_abort("VK generation not supported in WASM");
130#endif
131
132 const size_t total_cpus = get_num_cpus();
133 const size_t num_functions = functions.size();
134
135 // Heuristic for nested parallelism:
136 // - actual_tasks = min(num_functions, total_cpus)
137 // - threads_per_task = min(total_cpus, max(2, total_cpus / actual_tasks * 2))
138 size_t actual_tasks = std::min(num_functions, total_cpus);
139 size_t threads_per_task = std::min(total_cpus, std::max(size_t{ 2 }, total_cpus / actual_tasks * 2));
140
141 // Track work distribution
142 std::atomic<size_t> current_function{ 0 };
143
144 // Worker function
145 auto worker = [&]() {
146 // Set thread-local concurrency for this worker
147 set_parallel_for_concurrency(threads_per_task);
148
149 // Process functions
150 size_t func_idx;
151 while ((func_idx = current_function.fetch_add(1)) < num_functions) {
152 auto* function = functions[func_idx];
153 std::string fn_name = (*function)["name"].get<std::string>();
154
155 // Get bytecode from function
156 auto bytecode = extract_bytecode(*function);
157
158 // Generate and cache VK (can use parallel_for internally)
159 get_or_generate_cached_vk(cache_dir, fn_name, bytecode, force);
160 }
161 };
162
163 // Spawn threads
164 std::vector<std::thread> threads;
165 threads.reserve(actual_tasks);
166
167 for (size_t i = 0; i < actual_tasks; ++i) {
168 threads.emplace_back(worker);
169 }
170
171 // Wait for completion
172 for (auto& t : threads) {
173 t.join();
174 }
175
176 // Update JSON with VKs from cache (sequential is fine here, it's fast)
177 for (auto* function : functions) {
178 std::string fn_name = (*function)["name"].get<std::string>();
179
180 // Get bytecode to compute hash
181 auto bytecode = extract_bytecode(*function);
182
183 // Read VK from cache
184 std::string hash_str = compute_bytecode_hash(bytecode);
185 std::filesystem::path vk_cache_path = cache_dir / (hash_str + ".vk");
186 auto vk_data = read_file(vk_cache_path);
187
188 // Encode to base64 and store in JSON
189 std::string encoded_vk = base64_encode(vk_data.data(), vk_data.size(), false);
190 (*function)["verification_key"] = encoded_vk;
191 }
192}
193
194} // anonymous namespace
195
199bool transpile_artifact([[maybe_unused]] const std::string& input_path, [[maybe_unused]] const std::string& output_path)
200{
201#ifdef ENABLE_AVM_TRANSPILER
202 info("Transpiling: ", input_path, " -> ", output_path);
203
204 auto result = avm_transpile_file(input_path.c_str(), output_path.c_str());
205
206 if (result.success == 0) {
207 if (result.error_message) {
208 std::string error_msg(result.error_message);
209 if (error_msg == "Contract already transpiled") {
210 // Already transpiled, copy if different paths
211 if (input_path != output_path) {
212 std::filesystem::copy_file(
213 input_path, output_path, std::filesystem::copy_options::overwrite_existing);
214 }
215 } else {
216 info("Transpilation failed: ", error_msg);
217 avm_free_result(&result);
218 return false;
219 }
220 } else {
221 info("Transpilation failed");
222 avm_free_result(&result);
223 return false;
224 }
225 }
226
227 avm_free_result(&result);
228
229 info("Transpiled: ", input_path, " -> ", output_path);
230#else
231 throw_or_abort("AVM Transpiler is not enabled. Please enable it to use bb aztec_process.");
232#endif
233 return true;
234}
235
236bool process_aztec_artifact(const std::string& input_path, const std::string& output_path, bool force)
237{
238 if (!transpile_artifact(input_path, output_path)) {
239 return false;
240 }
241
242 // Verify output exists
243 if (!std::filesystem::exists(output_path)) {
244 throw_or_abort("Output file does not exist after transpilation");
245 }
246
247 // Step 2: Generate verification keys
248 auto cache_dir = get_cache_dir();
249 info("Generating verification keys for functions in ", std::filesystem::path(output_path).filename().string());
250 info("Cache directory: ", cache_dir.string());
251
252 // Read and parse artifact JSON
253 auto artifact_content = read_file(output_path);
254 std::string artifact_str(artifact_content.begin(), artifact_content.end());
255 auto artifact_json = nlohmann::json::parse(artifact_str);
256
257 if (!artifact_json.contains("functions")) {
258 info("Warning: No functions found in artifact");
259 return true;
260 }
261
262 // Filter to private constrained functions
263 std::vector<nlohmann::json*> private_functions;
264 for (auto& function : artifact_json["functions"]) {
265 if (is_private_constrained_function(function)) {
266 private_functions.push_back(&function);
267 }
268 }
269
270 if (private_functions.empty()) {
271 info("No private constrained functions found");
272 return true;
273 }
274
275 // Generate VKs
276 generate_vks_for_functions(cache_dir, private_functions, force);
277
278 // Write updated JSON back to file
279 std::ofstream out_file(output_path);
280 out_file << artifact_json.dump(2) << std::endl;
281 out_file.close();
282
283 info("Successfully processed: ", input_path, " -> ", output_path);
284 return true;
285}
286
287std::vector<std::string> find_contract_artifacts(const std::string& search_path)
288{
289 std::vector<std::string> artifacts;
290
291 // Recursively search for .json files in target/ directories, excluding cache/
292 for (const auto& entry : std::filesystem::recursive_directory_iterator(search_path)) {
293 if (!entry.is_regular_file()) {
294 continue;
295 }
296
297 const auto& path = entry.path();
298
299 // Must be a .json file
300 if (path.extension() != ".json") {
301 continue;
302 }
303
304 // Must be in a target/ directory
305 std::string path_str = path.string();
306 if (path_str.find("/target/") == std::string::npos && path_str.find("\\target\\") == std::string::npos) {
307 continue;
308 }
309
310 // Exclude cache directories and function artifact temporaries
311 if (path_str.find("/cache/") != std::string::npos || path_str.find("\\cache\\") != std::string::npos ||
312 path_str.find(".function_artifact_") != std::string::npos) {
313 continue;
314 }
315
316 artifacts.push_back(path.string());
317 }
318
319 return artifacts;
320}
321
322bool process_all_artifacts(const std::string& search_path, bool force)
323{
324 auto artifacts = find_contract_artifacts(search_path);
325
326 if (artifacts.empty()) {
327 info("No contract artifacts found. Please compile your contracts first with 'nargo compile'.");
328 return false;
329 }
330
331 info("Found ", artifacts.size(), " contract artifact(s) to process");
332
333 bool all_success = true;
334 for (const auto& artifact : artifacts) {
335 // Process in-place (input == output)
336 if (!process_aztec_artifact(artifact, artifact, force)) {
337 all_success = false;
338 }
339 }
340
341 if (all_success) {
342 info("Contract postprocessing complete!");
343 }
344
345 return all_success;
346}
347
348bool get_cache_paths(const std::string& input_path)
349{
350 try {
351 // Verify input exists
352 if (!std::filesystem::exists(input_path)) {
353 throw_or_abort("Input file does not exist: " + input_path);
354 }
355
356 // Read and parse artifact JSON
357 auto artifact_content = read_file(input_path);
358 std::string artifact_str(artifact_content.begin(), artifact_content.end());
359 auto artifact_json = nlohmann::json::parse(artifact_str);
360
361 if (!artifact_json.contains("functions")) {
362 // No functions, but not an error
363 return true;
364 }
365
366 // Get cache directory
367 auto cache_dir = get_cache_dir();
368
369 // Find all private constrained functions and output their cache paths
370 for (const auto& function : artifact_json["functions"]) {
371 if (!is_private_constrained_function(function)) {
372 continue;
373 }
374
375 std::string fn_name = function["name"].get<std::string>();
376 auto bytecode = extract_bytecode(function);
377 std::string hash_str = compute_bytecode_hash(bytecode);
378 std::filesystem::path vk_cache_path = cache_dir / (hash_str + ".vk");
379
380 // Output format: hash:cache_path:function_name
381 std::cout << hash_str << ":" << vk_cache_path.string() << ":" << fn_name << std::endl;
382 }
383
384 return true;
385 } catch (const std::exception& e) {
386 info("Error getting cache paths: ", e.what());
387 return false;
388 }
389}
390
391} // namespace bb
392#endif
std::shared_ptr< Napi::ThreadSafeFunction > bytecode
std::string base64_encode(unsigned char const *bytes_to_encode, size_t in_len, bool url)
Definition base64.cpp:117
Chonk-specific command definitions for the Barretenberg RPC API.
void info(Args... args)
Definition log.hpp:75
std::vector< uint8_t > decode_bytecode(const std::string &base64_bytecode)
void hash(State &state) noexcept
Sha256Hash sha256(const ByteContainer &input)
SHA-256 hash function (FIPS 180-4)
Definition sha256.cpp:150
Entry point for Barretenberg command-line interface.
Definition api.hpp:5
bool transpile_artifact(const std::string &input_path, const std::string &output_path)
Transpile the artifact file (or copy if transpiler not enabled)
bool process_all_artifacts(const std::string &search_path, bool force)
Process all discovered contract artifacts in a directory tree.
bool get_cache_paths(const std::string &input_path)
Get cache paths for all verification keys in an artifact.
bool process_aztec_artifact(const std::string &input_path, const std::string &output_path, bool force)
Process Aztec contract artifacts: transpile and generate verification keys.
size_t get_num_cpus()
Definition thread.cpp:33
std::vector< std::string > find_contract_artifacts(const std::string &search_path)
Find all contract artifacts in target/ directories.
const char * BB_VERSION
Definition version.hpp:14
std::vector< uint8_t > read_file(const std::string &filename, size_t bytes=0)
Definition file_io.hpp:29
void set_parallel_for_concurrency(size_t num_cores)
Definition thread.cpp:23
void write_file(const std::string &filename, std::vector< uint8_t > const &data)
Definition file_io.hpp:58
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
Definition tuple.hpp:13
void throw_or_abort(std::string const &err)