From 303af8661cb37f7a518e1c08518c8b66908186f3 Mon Sep 17 00:00:00 2001 From: Simon Hartcher Date: Thu, 13 Mar 2025 14:32:03 +1100 Subject: [PATCH 1/8] fix: make modules depend on codegen --- build.zig | 139 +++++++++++++++++++++++-------------------- src/servicemodel.zig | 2 +- 2 files changed, 75 insertions(+), 66 deletions(-) diff --git a/build.zig b/build.zig index 9de9394..8388869 100644 --- a/build.zig +++ b/build.zig @@ -64,17 +64,6 @@ pub fn build(b: *Builder) !void { const smithy_module = smithy_dep.module("smithy"); exe.root_module.addImport("smithy", smithy_module); // not sure this should be here... - // Expose module to others - _ = b.addModule("aws", .{ - .root_source_file = b.path("src/aws.zig"), - .imports = &.{.{ .name = "smithy", .module = smithy_module }}, - }); - - // Expose module to others - _ = b.addModule("aws-signing", .{ - .root_source_file = b.path("src/aws_signing.zig"), - .imports = &.{.{ .name = "smithy", .module = smithy_module }}, - }); // TODO: This does not work correctly due to https://github.com/ziglang/zig/issues/16354 // // We are working here with kind of a weird dependency though. So we can do this @@ -97,61 +86,81 @@ pub fn build(b: *Builder) !void { const run_step = b.step("run", "Run the app"); run_step.dependOn(&run_cmd.step); - const gen_step = blk: { - const cg = b.step("gen", "Generate zig service code from smithy models"); + const cg = b.step("gen", "Generate zig service code from smithy models"); - const cg_exe = b.addExecutable(.{ - .name = "codegen", - .root_source_file = b.path("codegen/src/main.zig"), - // We need this generated for the host, not the real target - .target = b.graph.host, - .optimize = if (b.verbose) .Debug else .ReleaseSafe, - }); - cg_exe.root_module.addImport("smithy", smithy_dep.module("smithy")); - var cg_cmd = b.addRunArtifact(cg_exe); - cg_cmd.addArg("--models"); - const hash = hash_blk: { - for (b.available_deps) |dep| { - const dep_name = dep.@"0"; - const dep_hash = dep.@"1"; - if (std.mem.eql(u8, dep_name, "models")) - break :hash_blk dep_hash; - } - return error.DependencyNamedModelsNotFoundInBuildZigZon; - }; - cg_cmd.addArg(try std.fs.path.join( - b.allocator, - &[_][]const u8{ - b.graph.global_cache_root.path.?, - "p", - hash, - models_subdir, - }, - )); - cg_cmd.addArg("--output"); - cg_cmd.addDirectoryArg(b.path("src/models")); - if (b.verbose) - cg_cmd.addArg("--verbose"); - // cg_cmd.step.dependOn(&fetch_step.step); - // TODO: this should use zig_exe from std.Build - // codegen should store a hash in a comment - // this would be hash of the exe that created the file - // concatenated with hash of input json. this would - // allow skipping generated files. May not include hash - // of contents of output file as maybe we want to tweak - // manually?? - // - // All the hashes can be in service_manifest.zig, which - // could be fun to just parse and go nuts. Top of - // file, generator exe hash. Each import has comment - // with both input and output hash and we can decide - // later about warning on manual changes... - - cg.dependOn(&cg_cmd.step); - break :blk cg; + const cg_exe = b.addExecutable(.{ + .name = "codegen", + .root_source_file = b.path("codegen/src/main.zig"), + // We need this generated for the host, not the real target + .target = b.graph.host, + .optimize = if (b.verbose) .Debug else .ReleaseSafe, + }); + cg_exe.root_module.addImport("smithy", smithy_dep.module("smithy")); + var cg_cmd = b.addRunArtifact(cg_exe); + cg_cmd.addArg("--models"); + const hash = hash_blk: { + for (b.available_deps) |dep| { + const dep_name = dep.@"0"; + const dep_hash = dep.@"1"; + if (std.mem.eql(u8, dep_name, "models")) + break :hash_blk dep_hash; + } + return error.DependencyNamedModelsNotFoundInBuildZigZon; }; + cg_cmd.addArg(try std.fs.path.join( + b.allocator, + &[_][]const u8{ + b.graph.global_cache_root.path.?, + "p", + hash, + models_subdir, + }, + )); + cg_cmd.addArg("--output"); + const cg_output_dir = cg_cmd.addOutputDirectoryArg("src/models"); + if (b.verbose) + cg_cmd.addArg("--verbose"); + // cg_cmd.step.dependOn(&fetch_step.step); + // TODO: this should use zig_exe from std.Build + // codegen should store a hash in a comment + // this would be hash of the exe that created the file + // concatenated with hash of input json. this would + // allow skipping generated files. May not include hash + // of contents of output file as maybe we want to tweak + // manually?? + // + // All the hashes can be in service_manifest.zig, which + // could be fun to just parse and go nuts. Top of + // file, generator exe hash. Each import has comment + // with both input and output hash and we can decide + // later about warning on manual changes... - exe.step.dependOn(gen_step); + cg.dependOn(&cg_cmd.step); + + exe.step.dependOn(cg); + + // This allows us to have each module depend on the + // generated service manifest. + const service_manifest_module = b.createModule(.{ + .root_source_file = cg_output_dir.path(b, "service_manifest.zig"), + .target = target, + .optimize = optimize, + }); + + // Expose module to others + _ = b.addModule("aws", .{ + .root_source_file = b.path("src/aws.zig"), + .imports = &.{ + .{ .name = "smithy", .module = smithy_module }, + .{ .name = "service_manifest", .module = service_manifest_module }, + }, + }); + + // Expose module to others + _ = b.addModule("aws-signing", .{ + .root_source_file = b.path("src/aws_signing.zig"), + .imports = &.{.{ .name = "smithy", .module = smithy_module }}, + }); // Similar to creating the run step earlier, this exposes a `test` step to // the `zig build --help` menu, providing a way for the user to request @@ -182,7 +191,7 @@ pub fn build(b: *Builder) !void { .optimize = optimize, }); unit_tests.root_module.addImport("smithy", smithy_dep.module("smithy")); - unit_tests.step.dependOn(gen_step); + unit_tests.step.dependOn(cg); const run_unit_tests = b.addRunArtifact(unit_tests); run_unit_tests.skip_foreign_checks = true; @@ -205,7 +214,7 @@ pub fn build(b: *Builder) !void { .optimize = optimize, }); smoke_test.root_module.addImport("smithy", smithy_dep.module("smithy")); - smoke_test.step.dependOn(gen_step); + smoke_test.step.dependOn(cg); const run_smoke_test = b.addRunArtifact(smoke_test); diff --git a/src/servicemodel.zig b/src/servicemodel.zig index 0f26886..9c8a8c7 100644 --- a/src/servicemodel.zig +++ b/src/servicemodel.zig @@ -1,5 +1,5 @@ const std = @import("std"); -const service_list = @import("models/service_manifest.zig"); +const service_list = @import("service_manifest"); const expectEqualStrings = std.testing.expectEqualStrings; pub fn Services(comptime service_imports: anytype) type { From 71495a4d1d7000815ba44e8d261ecd42d893a0de Mon Sep 17 00:00:00 2001 From: Simon Hartcher Date: Thu, 13 Mar 2025 14:46:15 +1100 Subject: [PATCH 2/8] chore: add codegen to paths --- build.zig.zon | 1 + 1 file changed, 1 insertion(+) diff --git a/build.zig.zon b/build.zig.zon index f678894..468f17d 100644 --- a/build.zig.zon +++ b/build.zig.zon @@ -5,6 +5,7 @@ "build.zig", "build.zig.zon", "src", + "codegen", "README.md", "LICENSE", }, From 220d45ab2075316216cfad824fc67ebbae84a9cd Mon Sep 17 00:00:00 2001 From: Simon Hartcher Date: Thu, 13 Mar 2025 15:43:32 +1100 Subject: [PATCH 3/8] fix: missing module imports --- build.zig | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/build.zig b/build.zig index 8388869..63bbf64 100644 --- a/build.zig +++ b/build.zig @@ -95,7 +95,7 @@ pub fn build(b: *Builder) !void { .target = b.graph.host, .optimize = if (b.verbose) .Debug else .ReleaseSafe, }); - cg_exe.root_module.addImport("smithy", smithy_dep.module("smithy")); + cg_exe.root_module.addImport("smithy", smithy_module); var cg_cmd = b.addRunArtifact(cg_exe); cg_cmd.addArg("--models"); const hash = hash_blk: { @@ -146,6 +146,9 @@ pub fn build(b: *Builder) !void { .target = target, .optimize = optimize, }); + service_manifest_module.addImport("smithy", smithy_module); + + exe.root_module.addImport("service_manifest", service_manifest_module); // Expose module to others _ = b.addModule("aws", .{ @@ -190,7 +193,8 @@ pub fn build(b: *Builder) !void { .target = b.resolveTargetQuery(t), .optimize = optimize, }); - unit_tests.root_module.addImport("smithy", smithy_dep.module("smithy")); + unit_tests.root_module.addImport("smithy", smithy_module); + unit_tests.root_module.addImport("service_manifest", service_manifest_module); unit_tests.step.dependOn(cg); const run_unit_tests = b.addRunArtifact(unit_tests); @@ -213,7 +217,8 @@ pub fn build(b: *Builder) !void { .target = target, .optimize = optimize, }); - smoke_test.root_module.addImport("smithy", smithy_dep.module("smithy")); + smoke_test.root_module.addImport("smithy", smithy_module); + smoke_test.root_module.addImport("service_manifest", service_manifest_module); smoke_test.step.dependOn(cg); const run_smoke_test = b.addRunArtifact(smoke_test); From cdaf92486734a5627b62e6199769c6c1d83ea4c0 Mon Sep 17 00:00:00 2001 From: Emil Lerch Date: Thu, 20 Mar 2025 19:57:14 -0700 Subject: [PATCH 4/8] update readme generally and for new branch strategy --- README.md | 59 +++++++++++++++++++++++++++++++++---------------------- 1 file changed, 35 insertions(+), 24 deletions(-) diff --git a/README.md b/README.md index cfb703a..73fef11 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,9 @@ AWS SDK for Zig =============== -[Zig 0.13](https://ziglang.org/download/#release-0.13.0): +[Zig 0.14](https://ziglang.org/download/#release-0.14.0): -[![Build Status: Zig 0.13.0](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/build.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=build.yaml&state=closed) +[![Build Status: Zig 0.14.0](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/build.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=build.yaml&state=closed) [Last Mach Nominated Zig Version](https://machengine.org/about/nominated-zig/): @@ -13,12 +13,13 @@ AWS SDK for Zig [![Build Status: Zig Nightly](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/zig-nightly.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=zig-nightly.yaml&state=closed) -**NOTE ON BUILD STATUS**: The nightly/mach nominated version of this currently -panics under CI, but I have not yet reproduced this panic. Running manually on -multiple machines appears to be working properly +[Zig 0.13](https://ziglang.org/download/#release-0.13.0): + +[![Build Status: Zig 0.13.0](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/zig-previous.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=zig-previous.yaml&state=closed) + Current executable size for the demo is 980k after compiling with -Doptimize=ReleaseSmall -in x86_linux, and will vary based on services used. Tested targets: +in x86_64-linux, and will vary based on services used. Tested targets: * x86_64-linux * riscv64-linux @@ -30,22 +31,38 @@ in x86_linux, and will vary based on services used. Tested targets: Tested targets are built, but not continuously tested, by CI. -Zig-Develop Branch ------------------- +Branches +-------- -This branch is intended for use with the in-development version of Zig. This -starts with 0.12.0-dev.3180+83e578a18. This is aligned with [Mach Engine's Nominated -Zig Versions](https://machengine.org/about/nominated-zig/). Nightly zig versions -are difficult to keep up with and there is no special effort made there, build -status is FYI (and used as a canary for nominated zig versions). +* **master**: This branch tracks the latest released zig version +* **zig-0.13**: This branch tracks the previous released zig version (0.13 currently). + Support for the previous version is best effort, generally + degrading over time. Fixes will generally appear in master, then + backported into the previous version. +* **zig-mach**: This branch tracks the latest mach nominated version. A separate + branch is necessary as mach nominated is usually, but not always, + more recent than the latest production zig. Support for the mach + version is best effort. +* **zig-develop**: This branch tracks zig nightly, and is used mainly as a canary + for breaking changes that will need to be dealt with when + a new mach nominated version or new zig release appear. + Expect significant delays in any build failures. + +Other branches/tags exist but are unsupported Building -------- `zig build` should work. It will build the code generation project, fetch model files from upstream AWS Go SDK v2, run the code generation, then build the main -project with the generated code. Testing can be done with `zig test`. +project with the generated code. Testing can be done with `zig build test`. Note that +this command tests on all supported architectures, so for a faster testing +process, use `zig build smoke-test` instead. +To make development even faster, a build option is provided to avoid the use of +LLVM. To use this, use the command `zig build -Dno-llvm smoke-test`. This +can reduce build/test time 300%. Note, however, native code generation in zig +is not yet complete, so you may see errors. Using ----- @@ -53,7 +70,8 @@ Using This is designed for use with the Zig package manager, and exposes a module called "aws". Set up `build.zig.zon` and add the dependency/module to your project as normal and the package manager should do its thing. A full example can be found -in [/example](example/README.md). +in [/example](example/build.zig.zon). This can also be used at build time in +a downstream project's `build.zig`. Configuring the module and/or Running the demo ---------------------------------------------- @@ -61,8 +79,8 @@ Configuring the module and/or Running the demo This library mimics the aws c libraries for it's work, so it operates like most other 'AWS things'. [/src/main.zig](src/main.zig) gives you a handful of examples for working with services. For local testing or alternative endpoints, there's -no real standard, so there is code to look for `AWS_ENDPOINT_URL` environment -variable that will supersede all other configuration. +no real standard, so there is code to look for an environment variable +`AWS_ENDPOINT_URL` variable that will supersede all other configuration. Limitations ----------- @@ -83,13 +101,6 @@ TODO List: * Implement timeouts and other TODO's in the code * Add option to cache signature keys -Services without TLS 1.3 support --------------------------------- - -All AWS services should support TLS 1.3 at this point, but there are many regions -and several partitions, and not all of them have been tested, so your mileage -may vary. If something doesn't work, please submit an issue to let others know. - Dependency tree --------------- From ffe3941dbe1283b14d34c617a47833b03c236da0 Mon Sep 17 00:00:00 2001 From: Emil Lerch Date: Thu, 20 Mar 2025 23:24:38 -0700 Subject: [PATCH 5/8] allow workflow dispatch on zig previous --- .gitea/workflows/zig-previous.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitea/workflows/zig-previous.yaml b/.gitea/workflows/zig-previous.yaml index ce27346..1bc831f 100644 --- a/.gitea/workflows/zig-previous.yaml +++ b/.gitea/workflows/zig-previous.yaml @@ -1,5 +1,6 @@ name: AWS-Zig Build on: + workflow_dispatch: push: branches: - 'zig-0.13' From 34c097e45f7edce141d68f7c0a5881a3a6e37e7c Mon Sep 17 00:00:00 2001 From: Emil Lerch Date: Thu, 20 Mar 2025 23:28:38 -0700 Subject: [PATCH 6/8] update nominated zig url --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 73fef11..7772041 100644 --- a/README.md +++ b/README.md @@ -5,7 +5,7 @@ AWS SDK for Zig [![Build Status: Zig 0.14.0](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/build.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=build.yaml&state=closed) -[Last Mach Nominated Zig Version](https://machengine.org/about/nominated-zig/): +[Last Mach Nominated Zig Version](https://machengine.org/docs/nominated-zig/): [![Build Status: Mach nominated](https://git.lerch.org/lobo/aws-sdk-for-zig/actions/workflows/zig-mach.yaml/badge.svg)](https://git.lerch.org/lobo/aws-sdk-for-zig/actions?workflow=zig-mach.yaml&state=closed) From 9e8b3a6fc60ce5bb8a8c8e1e9e1e65360d0a25ee Mon Sep 17 00:00:00 2001 From: Emil Lerch Date: Fri, 21 Mar 2025 09:59:33 -0700 Subject: [PATCH 7/8] fix json serialization for null/empty maps --- codegen/src/json.zig | 20 +++++++++++++++-- src/aws.zig | 52 ++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 70 insertions(+), 2 deletions(-) diff --git a/codegen/src/json.zig b/codegen/src/json.zig index f3f4f47..3fe93ec 100644 --- a/codegen/src/json.zig +++ b/codegen/src/json.zig @@ -4,7 +4,7 @@ const std = @import("std"); pub fn serializeMap(map: anytype, key: []const u8, options: anytype, out_stream: anytype) !bool { if (@typeInfo(@TypeOf(map)) == .optional) { if (map == null) - return true + return false else return serializeMapInternal(map.?, key, options, out_stream); } @@ -12,7 +12,23 @@ pub fn serializeMap(map: anytype, key: []const u8, options: anytype, out_stream: } fn serializeMapInternal(map: anytype, key: []const u8, options: anytype, out_stream: anytype) !bool { - if (map.len == 0) return true; + if (map.len == 0) { + var child_options = options; + if (child_options.whitespace) |*child_ws| + child_ws.indent_level += 1; + + try out_stream.writeByte('"'); + try out_stream.writeAll(key); + _ = try out_stream.write("\":"); + if (options.whitespace) |ws| { + if (ws.separator) { + try out_stream.writeByte(' '); + } + } + try out_stream.writeByte('{'); + try out_stream.writeByte('}'); + return true; + } // TODO: Map might be [][]struct{key, value} rather than []struct{key, value} var child_options = options; if (child_options.whitespace) |*child_ws| diff --git a/src/aws.zig b/src/aws.zig index 424e7b2..2738a1e 100644 --- a/src/aws.zig +++ b/src/aws.zig @@ -1310,6 +1310,58 @@ test "custom serialization for map objects" { , buffer.items); } +test "proper serialization for kms" { + // Github issue #8 + // https://github.com/elerch/aws-sdk-for-zig/issues/8 + const allocator = std.testing.allocator; + var buffer = std.ArrayList(u8).init(allocator); + defer buffer.deinit(); + const req = services.kms.encrypt.Request{ + .encryption_algorithm = "SYMMETRIC_DEFAULT", + // Since encryption_context is not null, we expect "{}" to be the value + // here, not "[]", because this is our special AWS map pattern + .encryption_context = &.{}, + .key_id = "42", + .plaintext = "foo", + .dry_run = false, + .grant_tokens = &[_][]const u8{}, + }; + try json.stringify(req, .{ .whitespace = .{} }, buffer.writer()); + try std.testing.expectEqualStrings( + \\{ + \\ "KeyId": "42", + \\ "Plaintext": "foo", + \\ "EncryptionContext": {}, + \\ "GrantTokens": [], + \\ "EncryptionAlgorithm": "SYMMETRIC_DEFAULT", + \\ "DryRun": false + \\} + , buffer.items); + + var buffer_null = std.ArrayList(u8).init(allocator); + defer buffer_null.deinit(); + const req_null = services.kms.encrypt.Request{ + .encryption_algorithm = "SYMMETRIC_DEFAULT", + // Since encryption_context here *IS* null, we expect simply "null" to be the value + .encryption_context = null, + .key_id = "42", + .plaintext = "foo", + .dry_run = false, + .grant_tokens = &[_][]const u8{}, + }; + try json.stringify(req_null, .{ .whitespace = .{} }, buffer_null.writer()); + try std.testing.expectEqualStrings( + \\{ + \\ "KeyId": "42", + \\ "Plaintext": "foo", + \\ "EncryptionContext": null, + \\ "GrantTokens": [], + \\ "EncryptionAlgorithm": "SYMMETRIC_DEFAULT", + \\ "DryRun": false + \\} + , buffer_null.items); +} + test "REST Json v1 builds proper queries" { const allocator = std.testing.allocator; const svs = Services(.{.lambda}){}; From e0e09fb19ef9176af45fb2beb41fc5017a7ea5f4 Mon Sep 17 00:00:00 2001 From: Emil Lerch Date: Sun, 23 Mar 2025 16:24:20 -0700 Subject: [PATCH 8/8] add no-bin option as recommended in zig 0.14.0 rlease notes https://ziglang.org/download/0.14.0/release-notes.html\#Incremental-Compilation --- build.zig | 20 ++++++++++++-------- 1 file changed, 12 insertions(+), 8 deletions(-) diff --git a/build.zig b/build.zig index 4cdb39a..812d699 100644 --- a/build.zig +++ b/build.zig @@ -34,12 +34,12 @@ pub fn build(b: *Builder) !void { "no-llvm", "Disable LLVM", ) orelse false; - const broken_windows = b.option( bool, "broken-windows", "Windows is broken in this environment (do not run Windows tests)", ) orelse false; + const no_bin = b.option(bool, "no-bin", "skip emitting binary") orelse false; // TODO: Embed the current git version in the code. We can do this // by looking for .git/HEAD (if it exists, follow the ref to /ref/heads/whatevs, // grab that commit, and use b.addOptions/exe.addOptions to generate the @@ -102,12 +102,12 @@ pub fn build(b: *Builder) !void { var cg_cmd = b.addRunArtifact(cg_exe); cg_cmd.addArg("--models"); cg_cmd.addArg(try std.fs.path.join( - b.allocator, - &[_][]const u8{ - try b.dependency("models", .{}).path("").getPath3(b, null).toString(b.allocator), - models_subdir, - }, - )); + b.allocator, + &[_][]const u8{ + try b.dependency("models", .{}).path("").getPath3(b, null).toString(b.allocator), + models_subdir, + }, + )); cg_cmd.addArg("--output"); const cg_output_dir = cg_cmd.addOutputDirectoryArg("src/models"); if (b.verbose) @@ -218,5 +218,9 @@ pub fn build(b: *Builder) !void { const run_smoke_test = b.addRunArtifact(smoke_test); smoke_test_step.dependOn(&run_smoke_test.step); - b.installArtifact(exe); + if (no_bin) { + b.getInstallStep().dependOn(&exe.step); + } else { + b.installArtifact(exe); + } }