add stubs for batchgetitem and batchwriteitem (parsing implemented in bwi)
This commit is contained in:
parent
1a53c3cf8a
commit
2eee4c5020
@ -1,5 +1,6 @@
|
||||
const std = @import("std");
|
||||
const encryption = @import("encryption.zig");
|
||||
const sqlite = @import("sqlite"); // TODO: If we use this across all services, Account should not have this, and we should have a localdbaccount struct
|
||||
|
||||
const test_account_key = "09aGW6z6QofVsPlWP9FGqVnshxHWAWrKZwLkwkgWs7w=";
|
||||
|
||||
@ -25,3 +26,46 @@ pub fn accountForId(allocator: std.mem.Allocator, account_id: []const u8) !Self
|
||||
pub fn deinit(self: Self) void {
|
||||
self.allocator.free(self.root_account_key);
|
||||
}
|
||||
|
||||
pub var data_dir: []const u8 = "";
|
||||
|
||||
/// Gets the database for this account. If under test, a memory database is used
|
||||
/// instead. Will initialize the database with appropriate metadata tables
|
||||
pub fn dbForAccount(allocator: std.mem.Allocator, account_id: []const u8) !sqlite.Db {
|
||||
// TODO: Need to move this function somewhere central
|
||||
// TODO: Need configuration for what directory to use
|
||||
// TODO: Should this be a pool, and if so, how would we know when to close?
|
||||
const file_without_path = try std.fmt.allocPrint(allocator, "ddb-{s}.sqlite3", .{account_id});
|
||||
defer allocator.free(file_without_path);
|
||||
const db_file_name = try std.fs.path.joinZ(allocator, &[_][]const u8{ data_dir, file_without_path });
|
||||
defer allocator.free(db_file_name);
|
||||
const mode = if (@import("builtin").is_test) sqlite.Db.Mode.Memory else sqlite.Db.Mode{ .File = db_file_name };
|
||||
const new = mode == .Memory or (std.fs.cwd().statFile(file_without_path) catch null == null);
|
||||
var db = try sqlite.Db.init(.{
|
||||
.mode = mode,
|
||||
.open_flags = .{
|
||||
.write = true,
|
||||
.create = new,
|
||||
},
|
||||
.threading_mode = .MultiThread,
|
||||
});
|
||||
|
||||
// DDB minimum table name length is 3. DDB local creates this table with metadata
|
||||
// This of course is only if the database is first run
|
||||
if (new)
|
||||
try db.exec(
|
||||
\\CREATE TABLE dm (
|
||||
\\ TableName TEXT,
|
||||
\\ CreationDateTime INTEGER,
|
||||
\\ LastDecreaseDate INTEGER,
|
||||
\\ LastIncreaseDate INTEGER,
|
||||
\\ NumberOfDecreasesToday INTEGER,
|
||||
\\ ReadCapacityUnits INTEGER,
|
||||
\\ WriteCapacityUnits INTEGER,
|
||||
\\ TableInfo BLOB,
|
||||
\\ BillingMode INTEGER DEFAULT 0,
|
||||
\\ PayPerRequestDateTime INTEGER DEFAULT 0,
|
||||
\\ PRIMARY KEY(TableName))
|
||||
, .{}, .{});
|
||||
return db;
|
||||
}
|
||||
|
17
src/batchgetitem.zig
Normal file
17
src/batchgetitem.zig
Normal file
@ -0,0 +1,17 @@
|
||||
const std = @import("std");
|
||||
const sqlite = @import("sqlite");
|
||||
const AuthenticatedRequest = @import("AuthenticatedRequest.zig");
|
||||
const Account = @import("Account.zig");
|
||||
const encryption = @import("encryption.zig");
|
||||
const returnException = @import("main.zig").returnException;
|
||||
|
||||
pub fn handler(request: *AuthenticatedRequest, writer: anytype) ![]const u8 {
|
||||
_ = writer;
|
||||
const allocator = request.allocator;
|
||||
const account_id = request.account_id;
|
||||
_ = account_id;
|
||||
|
||||
var parsed = try std.json.parseFromSlice(std.json.Value, allocator, request.event_data, .{});
|
||||
defer parsed.deinit();
|
||||
// const request_params = try parseRequest(request, parsed, writer);
|
||||
}
|
646
src/batchwriteitem.zig
Normal file
646
src/batchwriteitem.zig
Normal file
@ -0,0 +1,646 @@
|
||||
const std = @import("std");
|
||||
const sqlite = @import("sqlite");
|
||||
const AuthenticatedRequest = @import("AuthenticatedRequest.zig");
|
||||
const Account = @import("Account.zig");
|
||||
const encryption = @import("encryption.zig");
|
||||
const ddb_types = @import("ddb_types.zig");
|
||||
const returnException = @import("main.zig").returnException;
|
||||
|
||||
// Copied from ddb_type and made inferred. Yuck :(
|
||||
pub const AttributeTypeName = enum {
|
||||
string,
|
||||
number,
|
||||
binary,
|
||||
boolean,
|
||||
null,
|
||||
map,
|
||||
list,
|
||||
string_set,
|
||||
number_set,
|
||||
binary_set,
|
||||
};
|
||||
|
||||
// Cannot use AttributeTypeName enum as it is not inferred
|
||||
// const AttributeValue = union(ddb_types.AttributeTypeName) {
|
||||
const AttributeValue = union(AttributeTypeName) {
|
||||
string: []const u8,
|
||||
number: []const u8, // Floating point stored as string
|
||||
binary: []const u8, // Base64-encoded binary data object
|
||||
boolean: bool,
|
||||
null: bool,
|
||||
map: std.json.ObjectMap, // We're just holding the json...in the DB we probably just stringify this?
|
||||
// "M": {"Name": {"S": "Joe"}, "Age": {"N": "35"}}
|
||||
list: std.json.Array, // Again, just hoding json here:
|
||||
// "L": [ {"S": "Cookies"} , {"S": "Coffee"}, {"N": "3.14159"}]
|
||||
string_set: [][]const u8,
|
||||
number_set: [][]const u8,
|
||||
binary_set: [][]const u8,
|
||||
|
||||
const Self = @This();
|
||||
pub fn validate(self: Self) !void {
|
||||
switch (self) {
|
||||
.string, .string_set, .boolean, .null, .map, .list => {},
|
||||
.number => |s| _ = try std.fmt.parseFloat(f64, s),
|
||||
.binary => |s| try base64Validate(std.base64.standard.Decoder, s),
|
||||
.number_set => |ns| for (ns) |s| {
|
||||
_ = try std.fmt.parseFloat(f64, s);
|
||||
},
|
||||
.binary_set => |bs| for (bs) |s| try base64Validate(std.base64.standard.Decoder, s),
|
||||
}
|
||||
}
|
||||
|
||||
fn base64Validate(decoder: std.base64.Base64Decoder, source: []const u8) std.base64.Error!void {
|
||||
const invalid_char = 0xff;
|
||||
// This is taken from the stdlib decode function and modified to simply
|
||||
// not write anything
|
||||
if (decoder.pad_char != null and source.len % 4 != 0) return error.InvalidPadding;
|
||||
var acc: u12 = 0;
|
||||
var acc_len: u4 = 0;
|
||||
var leftover_idx: ?usize = null;
|
||||
for (source, 0..) |c, src_idx| {
|
||||
const d = decoder.char_to_index[c];
|
||||
if (d == invalid_char) {
|
||||
if (decoder.pad_char == null or c != decoder.pad_char.?) return error.InvalidCharacter;
|
||||
leftover_idx = src_idx;
|
||||
break;
|
||||
}
|
||||
acc = (acc << 6) + d;
|
||||
acc_len += 6;
|
||||
if (acc_len >= 8) {
|
||||
acc_len -= 8;
|
||||
}
|
||||
}
|
||||
if (acc_len > 4 or (acc & (@as(u12, 1) << acc_len) - 1) != 0) {
|
||||
return error.InvalidPadding;
|
||||
}
|
||||
if (leftover_idx == null) return;
|
||||
var leftover = source[leftover_idx.?..];
|
||||
if (decoder.pad_char) |pad_char| {
|
||||
const padding_len = acc_len / 2;
|
||||
var padding_chars: usize = 0;
|
||||
for (leftover) |c| {
|
||||
if (c != pad_char) {
|
||||
return if (c == invalid_char) error.InvalidCharacter else error.InvalidPadding;
|
||||
}
|
||||
padding_chars += 1;
|
||||
}
|
||||
if (padding_chars != padding_len) return error.InvalidPadding;
|
||||
}
|
||||
}
|
||||
};
|
||||
const Attribute = struct {
|
||||
name: []const u8,
|
||||
value: AttributeValue,
|
||||
};
|
||||
|
||||
const Request = struct {
|
||||
put_request: ?[]Attribute,
|
||||
delete_request: ?[]Attribute,
|
||||
};
|
||||
const RequestItem = struct {
|
||||
table_name: []const u8,
|
||||
requests: []Request,
|
||||
};
|
||||
|
||||
const ReturnConsumedCapacity = enum {
|
||||
indexes,
|
||||
total,
|
||||
none,
|
||||
};
|
||||
const Params = struct {
|
||||
request_items: []RequestItem,
|
||||
return_consumed_capacity: ReturnConsumedCapacity = .none,
|
||||
return_item_collection_metrics: bool = false,
|
||||
arena: *std.heap.ArenaAllocator,
|
||||
|
||||
pub fn deinit(self: *Params) void {
|
||||
const allocator = self.arena.child_allocator;
|
||||
self.arena.deinit();
|
||||
allocator.destroy(self.arena);
|
||||
}
|
||||
pub fn validate(self: Params) !void {
|
||||
for (self.request_items) |item| {
|
||||
for (item.requests) |request| {
|
||||
if (request.put_request) |put| {
|
||||
for (put) |attribute| try attribute.value.validate();
|
||||
}
|
||||
if (request.delete_request) |del| {
|
||||
for (del) |attribute| try attribute.value.validate();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
pub fn parseRequest(allocator: std.mem.Allocator, request: *AuthenticatedRequest, writer: anytype) !Params {
|
||||
// This pattern borrowed from https://ziglang.org/documentation/0.11.0/std/src/std/json/static.zig.html
|
||||
var rc = Params{
|
||||
.arena = try allocator.create(std.heap.ArenaAllocator),
|
||||
.request_items = undefined,
|
||||
};
|
||||
errdefer allocator.destroy(rc.arena);
|
||||
// I think the idea here is that we've created the allocator above, and this
|
||||
// line here rewrites the values (internal state) at the original pointer address
|
||||
rc.arena.* = std.heap.ArenaAllocator.init(allocator);
|
||||
errdefer rc.arena.deinit();
|
||||
var aa = rc.arena.allocator();
|
||||
|
||||
var parsed = try std.json.parseFromSliceLeaky(std.json.Value, aa, request.event_data, .{});
|
||||
|
||||
// RequestItems is most important, and is required. Check it first
|
||||
const ri = parsed.object.get("RequestItems");
|
||||
if (ri == null or ri.? != .object or ri.?.object.count() == 0)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Request missing RequestItems",
|
||||
);
|
||||
const request_items = ri.?.object.count();
|
||||
if (parsed.object.get("ReturnConsumedCapacity")) |rcc| {
|
||||
if (rcc != .string or
|
||||
(!std.mem.eql(u8, rcc.string, "INDEXES") and
|
||||
!std.mem.eql(u8, rcc.string, "TOTAL") and
|
||||
!std.mem.eql(u8, rcc.string, "NONE")))
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"ReturnConsumedCapacity value invalid. Valid values are INDEXES | TOTAL | NONE",
|
||||
);
|
||||
const val = try std.ascii.allocLowerString(aa, rcc.string);
|
||||
rc.return_consumed_capacity = std.meta.stringToEnum(ReturnConsumedCapacity, val).?;
|
||||
}
|
||||
if (parsed.object.get("ReturnItemCollectionMetrics")) |rcm| {
|
||||
if (rcm != .string or
|
||||
(!std.mem.eql(u8, rcm.string, "SIZE") and
|
||||
!std.mem.eql(u8, rcm.string, "NONE")))
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"ReturnCollectionMetrics value invalid. Valid values are SIZE | NONE",
|
||||
);
|
||||
rc.return_item_collection_metrics = std.mem.eql(u8, rcm.string, "SIZE");
|
||||
}
|
||||
// Good so far...let's allocate the request item array and process
|
||||
rc.request_items = try aa.alloc(RequestItem, request_items);
|
||||
var inx: usize = 0;
|
||||
var param_iterator = ri.?.object.iterator();
|
||||
while (param_iterator.next()) |p| : (inx += 1) {
|
||||
const key = p.key_ptr.*;
|
||||
const val = p.value_ptr.*;
|
||||
if (val != .array or val.array.items.len == 0)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"RequestItems object values must be non-zero length arrays",
|
||||
);
|
||||
|
||||
var request_item = try aa.create(RequestItem);
|
||||
// This arena doesn't deinit until after Params is done, so
|
||||
// we should be good to *NOT* duplicate these
|
||||
request_item.table_name = key; // try aa.dupe(key);
|
||||
request_item.requests = try aa.alloc(Request, val.array.items.len);
|
||||
rc.request_items[inx] = request_item.*;
|
||||
for (val.array.items, 0..) |req, jnx| {
|
||||
if (req != .object)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Non-object found in RequestItems array",
|
||||
);
|
||||
const request_key_count = req.object.count();
|
||||
if (request_key_count == 0)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Found unsupported request in RequestItems. Request has no PutRequest or DeleteRequest",
|
||||
);
|
||||
if (request_key_count > 2)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Found unsupported request in RequestItems. Too many keys",
|
||||
);
|
||||
|
||||
var table_request = try aa.create(Request);
|
||||
table_request.put_request = null;
|
||||
table_request.delete_request = null;
|
||||
var req_item_iterator = req.object.iterator();
|
||||
while (req_item_iterator.next()) |put_or_delete| {
|
||||
if (!std.mem.eql(u8, put_or_delete.key_ptr.*, "DeleteRequest") and
|
||||
!std.mem.eql(u8, put_or_delete.key_ptr.*, "PutRequest"))
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Found unsupported request in RequestItems. Valid requests are PutRequest and DeleteRequest",
|
||||
);
|
||||
if (put_or_delete.value_ptr.* != .object)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Request in RequestItems found with non-object value",
|
||||
);
|
||||
// We have a put or a delete with an object value. ok to proceed
|
||||
const is_put = std.mem.eql(u8, put_or_delete.key_ptr.*, "PutRequest");
|
||||
const pod_val = put_or_delete.value_ptr.*;
|
||||
if (is_put) {
|
||||
const put_val = pod_val.object.get("Item");
|
||||
if (put_val == null or put_val.? != .object)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"PutRequest in RequestItems found without Item object",
|
||||
);
|
||||
// Parse item object and assign to array
|
||||
table_request.put_request = try parseAttributes(aa, put_val.?.object, request, writer);
|
||||
} else {
|
||||
const del_val = pod_val.object.get("Keys");
|
||||
if (del_val == null or del_val.? != .object)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"DeleteRequest in RequestItems found without Key object",
|
||||
);
|
||||
// Parse key object and assign to array
|
||||
table_request.delete_request = try parseAttributes(aa, del_val.?.object, request, writer);
|
||||
}
|
||||
}
|
||||
rc.request_items[inx].requests[jnx] = table_request.*;
|
||||
}
|
||||
}
|
||||
return rc;
|
||||
// {
|
||||
// "RequestItems": {
|
||||
// "string" : [
|
||||
// {
|
||||
// "DeleteRequest": {
|
||||
// "Key": {
|
||||
// "string" : {
|
||||
// "B": blob,
|
||||
// "BOOL": boolean,
|
||||
// "BS": [ blob ],
|
||||
// "L": [
|
||||
// "AttributeValue"
|
||||
// ],
|
||||
// "M": {
|
||||
// "string" : "AttributeValue"
|
||||
// },
|
||||
// "N": "string",
|
||||
// "NS": [ "string" ],
|
||||
// "NULL": boolean,
|
||||
// "S": "string",
|
||||
// "SS": [ "string" ]
|
||||
// }
|
||||
// }
|
||||
// },
|
||||
// "PutRequest": {
|
||||
// "Item": {
|
||||
// "string" : {
|
||||
// "B": blob,
|
||||
// "BOOL": boolean,
|
||||
// "BS": [ blob ],
|
||||
// "L": [
|
||||
// "AttributeValue"
|
||||
// ],
|
||||
// "M": {
|
||||
// "string" : "AttributeValue"
|
||||
// },
|
||||
// "N": "string",
|
||||
// "NS": [ "string" ],
|
||||
// "NULL": boolean,
|
||||
// "S": "string",
|
||||
// "SS": [ "string" ]
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// ]
|
||||
// },
|
||||
// "ReturnConsumedCapacity": "string",
|
||||
// "ReturnItemCollectionMetrics": "string"
|
||||
// }
|
||||
}
|
||||
fn parseAttributes(
|
||||
arena: std.mem.Allocator,
|
||||
value: anytype,
|
||||
request: *AuthenticatedRequest,
|
||||
writer: anytype,
|
||||
) ![]Attribute {
|
||||
// {
|
||||
// "string" : {
|
||||
// "B": blob,
|
||||
// "BOOL": boolean,
|
||||
// "BS": [ blob ],
|
||||
// "L": [
|
||||
// "AttributeValue"
|
||||
// ],
|
||||
// "M": {
|
||||
// "string" : "AttributeValue"
|
||||
// },
|
||||
// "N": "string",
|
||||
// "NS": [ "string" ],
|
||||
// "NULL": boolean,
|
||||
// "S": "string",
|
||||
// "SS": [ "string" ]
|
||||
// }
|
||||
// }
|
||||
var attribute_count = value.count();
|
||||
if (attribute_count == 0)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Request in RequestItems found without any attributes in object",
|
||||
);
|
||||
var rc = try arena.alloc(Attribute, attribute_count);
|
||||
var iterator = value.iterator();
|
||||
var inx: usize = 0;
|
||||
while (iterator.next()) |att| : (inx += 1) {
|
||||
const key = att.key_ptr.*;
|
||||
const val = att.value_ptr.*;
|
||||
// std.debug.print(" \n====\nkey = \"{s}\"\nval = {any}\n====\n", .{ key, val.object.count() });
|
||||
if (val != .object or val.object.count() != 1)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Request in RequestItems found invalid attributes in object",
|
||||
);
|
||||
rc[inx].name = key; //try arena.dupe(u8, key);
|
||||
var val_iterator = val.object.iterator();
|
||||
var val_val = val_iterator.next().?;
|
||||
const attribute_type = val_val.key_ptr.*; // This should be "S", "N", "NULL", "BOOL", etc
|
||||
const attribute_value = val_val.value_ptr.*;
|
||||
// Convert this to our enum
|
||||
const attribute_type_enum = std.meta.stringToEnum(ddb_types.AttributeTypeDescriptor, attribute_type);
|
||||
if (attribute_type_enum == null)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Request in RequestItems found attribute with invalid type",
|
||||
);
|
||||
// Convert our enum to something that looks better when reading code
|
||||
const attribute_type_enum_converted = @as(ddb_types.AttributeTypeName, @enumFromInt(@intFromEnum(attribute_type_enum.?)));
|
||||
// Now we need to get *THIS* enum over to our union, which uses the same values
|
||||
// We'll just use a switch here, because each of these cases must
|
||||
// be handled slightly differently
|
||||
var final_attribute_value: AttributeValue = undefined;
|
||||
switch (attribute_type_enum_converted) {
|
||||
.string => {
|
||||
try expectType(attribute_value, .string, request, writer);
|
||||
final_attribute_value = .{ .string = attribute_value.string };
|
||||
},
|
||||
.number => {
|
||||
// There is a .number_string, but I think that is for stringify?
|
||||
try expectType(attribute_value, .string, request, writer);
|
||||
final_attribute_value = .{ .number = attribute_value.string };
|
||||
},
|
||||
.binary => {
|
||||
try expectType(attribute_value, .string, request, writer);
|
||||
final_attribute_value = .{ .binary = attribute_value.string };
|
||||
},
|
||||
.boolean => {
|
||||
try expectType(attribute_value, .bool, request, writer);
|
||||
final_attribute_value = .{ .boolean = attribute_value.bool };
|
||||
},
|
||||
.null => {
|
||||
try expectType(attribute_value, .bool, request, writer);
|
||||
final_attribute_value = .{ .null = attribute_value.bool };
|
||||
},
|
||||
.map => {
|
||||
try expectType(attribute_value, .object, request, writer);
|
||||
final_attribute_value = .{ .map = attribute_value.object };
|
||||
},
|
||||
.list => {
|
||||
try expectType(attribute_value, .array, request, writer);
|
||||
final_attribute_value = .{ .list = attribute_value.array };
|
||||
},
|
||||
.string_set => {
|
||||
try expectType(attribute_value, .array, request, writer);
|
||||
final_attribute_value = .{ .string_set = try toStringArray(arena, attribute_value.array, request, writer) };
|
||||
},
|
||||
.number_set => {
|
||||
try expectType(attribute_value, .array, request, writer);
|
||||
final_attribute_value = .{ .number_set = try toStringArray(arena, attribute_value.array, request, writer) };
|
||||
},
|
||||
.binary_set => {
|
||||
try expectType(attribute_value, .array, request, writer);
|
||||
final_attribute_value = .{ .binary_set = try toStringArray(arena, attribute_value.array, request, writer) };
|
||||
},
|
||||
}
|
||||
rc[inx].value = final_attribute_value;
|
||||
}
|
||||
return rc;
|
||||
}
|
||||
|
||||
fn toStringArray(
|
||||
arena: std.mem.Allocator,
|
||||
arr: std.json.Array,
|
||||
request: *AuthenticatedRequest,
|
||||
writer: anytype,
|
||||
) ![][]const u8 {
|
||||
var rc = try arena.alloc([]const u8, arr.items.len);
|
||||
for (arr.items, 0..) |item, inx| {
|
||||
try expectType(item, .string, request, writer);
|
||||
rc[inx] = item.string;
|
||||
}
|
||||
return rc;
|
||||
}
|
||||
|
||||
fn expectType(actual: std.json.Value, comptime expected: @TypeOf(.enum_literal), request: *AuthenticatedRequest, writer: anytype) !void {
|
||||
if (actual != expected)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Attribute type does not match expected type",
|
||||
);
|
||||
if (actual == .array and actual.array.items.len == 0)
|
||||
try returnException(
|
||||
request,
|
||||
.bad_request,
|
||||
error.ValidationException,
|
||||
writer,
|
||||
"Attribute array cannot be empty",
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
pub fn handler(request: *AuthenticatedRequest, writer: anytype) ![]const u8 {
|
||||
const allocator = request.allocator;
|
||||
const account_id = request.account_id;
|
||||
_ = account_id;
|
||||
|
||||
var params = try Params.parseRequest(allocator, request, writer);
|
||||
defer params.deinit();
|
||||
// 1. Get the list of encrypted table names using the account id root key
|
||||
// 2. Get the matching table-scope encryption keys
|
||||
// 3. For each table request:
|
||||
// 1. Find the hash values of put and delete requests in the request
|
||||
// 2. Encrypt the hash values
|
||||
// 3. Delete any existing records with that hash value (for delete requests, we're done here)
|
||||
// 4. If put request, put the new item in the table (with encrypted values, using table encryption)
|
||||
// TODO: Capacity limiting and metrics
|
||||
}
|
||||
|
||||
test "basic request parsing" {
|
||||
const allocator = std.testing.allocator;
|
||||
var request = AuthenticatedRequest{
|
||||
.output_format = .text,
|
||||
.event_data =
|
||||
\\ {
|
||||
\\ "RequestItems": {
|
||||
\\ "Forum": [
|
||||
\\ {
|
||||
\\ "PutRequest": {
|
||||
\\ "Item": {
|
||||
\\ "Name": {
|
||||
\\ "S": "Amazon DynamoDB"
|
||||
\\ },
|
||||
\\ "Category": {
|
||||
\\ "S": "Amazon Web Services"
|
||||
\\ }
|
||||
\\ }
|
||||
\\ }
|
||||
\\ }
|
||||
\\ ]
|
||||
\\ }
|
||||
\\ }
|
||||
,
|
||||
.headers = undefined,
|
||||
.status = .ok,
|
||||
.reason = "",
|
||||
.account_id = "1234",
|
||||
.allocator = allocator,
|
||||
};
|
||||
var al = std.ArrayList(u8).init(allocator);
|
||||
defer al.deinit();
|
||||
var writer = al.writer();
|
||||
var parms = try Params.parseRequest(allocator, &request, writer);
|
||||
defer parms.deinit();
|
||||
try std.testing.expect(parms.return_consumed_capacity == .none);
|
||||
try std.testing.expect(!parms.return_item_collection_metrics);
|
||||
try std.testing.expect(parms.request_items.len == 1);
|
||||
const forum = parms.request_items[0];
|
||||
try std.testing.expectEqualStrings("Forum", forum.table_name);
|
||||
try std.testing.expect(forum.requests.len == 1);
|
||||
const put_and_or_delete = forum.requests[0];
|
||||
try std.testing.expect(put_and_or_delete.put_request != null);
|
||||
const put = put_and_or_delete.put_request.?;
|
||||
try std.testing.expect(put.len == 2);
|
||||
const name = put[0];
|
||||
const category = put[1];
|
||||
try std.testing.expectEqualStrings("Name", name.name);
|
||||
try std.testing.expectEqualStrings("Category", category.name);
|
||||
try std.testing.expect(name.value == .string);
|
||||
try std.testing.expect(category.value == .string);
|
||||
try std.testing.expectEqualStrings("Amazon DynamoDB", name.value.string);
|
||||
try std.testing.expectEqualStrings("Amazon Web Services", category.value.string);
|
||||
try std.testing.expect(put_and_or_delete.delete_request == null);
|
||||
}
|
||||
test "all types request parsing" {
|
||||
const allocator = std.testing.allocator;
|
||||
var request = AuthenticatedRequest{
|
||||
.output_format = .text,
|
||||
.event_data =
|
||||
\\ {
|
||||
\\ "RequestItems": {
|
||||
\\ "Forum": [
|
||||
\\ {
|
||||
\\ "PutRequest": {
|
||||
\\ "Item": {
|
||||
\\ "String": {
|
||||
\\ "S": "Amazon DynamoDB"
|
||||
\\ },
|
||||
\\ "Number": {
|
||||
\\ "N": "1.3"
|
||||
\\ },
|
||||
\\ "Binary": {
|
||||
\\ "B": "dGhpcyB0ZXh0IGlzIGJhc2U2NC1lbmNvZGVk"
|
||||
\\ },
|
||||
\\ "Boolean": {
|
||||
\\ "BOOL": true
|
||||
\\ },
|
||||
\\ "Null": {
|
||||
\\ "NULL": true
|
||||
\\ },
|
||||
\\ "List": {
|
||||
\\ "L": [ {"S": "Cookies"} , {"S": "Coffee"}, {"N": "3.14159"}]
|
||||
\\ },
|
||||
\\ "Map": {
|
||||
\\ "M": {"Name": {"S": "Joe"}, "Age": {"N": "35"}}
|
||||
\\ },
|
||||
\\ "Number Set": {
|
||||
\\ "NS": ["42.2", "-19", "7.5", "3.14"]
|
||||
\\ },
|
||||
\\ "Binary Set": {
|
||||
\\ "BS": ["U3Vubnk=", "UmFpbnk=", "U25vd3k="]
|
||||
\\ },
|
||||
\\ "String Set": {
|
||||
\\ "SS": ["Giraffe", "Hippo" ,"Zebra"]
|
||||
\\ }
|
||||
\\ }
|
||||
\\ }
|
||||
\\ }
|
||||
\\ ]
|
||||
\\ }
|
||||
\\ }
|
||||
,
|
||||
.headers = undefined,
|
||||
.status = .ok,
|
||||
.reason = "",
|
||||
.account_id = "1234",
|
||||
.allocator = allocator,
|
||||
};
|
||||
var al = std.ArrayList(u8).init(allocator);
|
||||
defer al.deinit();
|
||||
var writer = al.writer();
|
||||
var parms = try Params.parseRequest(allocator, &request, writer);
|
||||
defer parms.deinit();
|
||||
try parms.validate();
|
||||
try std.testing.expect(parms.return_consumed_capacity == .none);
|
||||
try std.testing.expect(!parms.return_item_collection_metrics);
|
||||
try std.testing.expect(parms.request_items.len == 1);
|
||||
const forum = parms.request_items[0];
|
||||
try std.testing.expectEqualStrings("Forum", forum.table_name);
|
||||
try std.testing.expect(forum.requests.len == 1);
|
||||
const put_and_or_delete = forum.requests[0];
|
||||
try std.testing.expect(put_and_or_delete.put_request != null);
|
||||
const put = put_and_or_delete.put_request.?;
|
||||
try std.testing.expect(put.len == 10);
|
||||
try std.testing.expectEqualStrings("String", put[0].name);
|
||||
try std.testing.expectEqualStrings("Number", put[1].name);
|
||||
try std.testing.expect(put[1].value == .number);
|
||||
try std.testing.expectEqualStrings("Binary", put[2].name);
|
||||
try std.testing.expect(put[2].value == .binary);
|
||||
try std.testing.expect(put[2].value.binary.len > 0);
|
||||
var buf = try allocator.alloc(u8, "this text is base64-encoded".len);
|
||||
defer allocator.free(buf);
|
||||
try std.base64.standard.Decoder.decode(buf, put[2].value.binary);
|
||||
try std.testing.expectEqualStrings("this text is base64-encoded", buf);
|
||||
try std.testing.expect(put_and_or_delete.delete_request == null);
|
||||
}
|
@ -3,7 +3,8 @@ const sqlite = @import("sqlite");
|
||||
const AuthenticatedRequest = @import("AuthenticatedRequest.zig");
|
||||
const Account = @import("Account.zig");
|
||||
const encryption = @import("encryption.zig");
|
||||
pub var data_dir: []const u8 = "";
|
||||
const returnException = @import("main.zig").returnException;
|
||||
const ddb_types = @import("ddb_types.zig");
|
||||
|
||||
// These are in the original casing so as to make the error messages nice
|
||||
const RequiredFields = enum(u3) {
|
||||
@ -14,38 +15,8 @@ const RequiredFields = enum(u3) {
|
||||
// zig fmt: on
|
||||
};
|
||||
|
||||
const AttributeTypeDescriptor = enum(u4) {
|
||||
S = 0,
|
||||
N = 1,
|
||||
B = 2,
|
||||
BOOL = 3,
|
||||
NULL = 4,
|
||||
M = 5,
|
||||
L = 6,
|
||||
SS = 7,
|
||||
NS = 8,
|
||||
BS = 9,
|
||||
};
|
||||
|
||||
const AttributeTypeName = enum(4) {
|
||||
String = 0,
|
||||
Number = 1,
|
||||
Binary = 2,
|
||||
Boolean = 3,
|
||||
Null = 4,
|
||||
Map = 5,
|
||||
List = 6,
|
||||
StringSet = 7,
|
||||
NumberSet = 8,
|
||||
BinarySet = 9,
|
||||
};
|
||||
|
||||
const AttributeDefinition = struct {
|
||||
name: []const u8,
|
||||
type: AttributeTypeDescriptor,
|
||||
};
|
||||
const TableInfo = struct {
|
||||
attribute_definitions: []*AttributeDefinition,
|
||||
attribute_definitions: []*ddb_types.AttributeDefinition,
|
||||
// gsi_list: []const u8, // Not sure how this is used
|
||||
// gsi_description_list: []const u8, // Not sure how this is used
|
||||
// sqlite_index: []const u8, // Not sure how this is used
|
||||
@ -73,7 +44,7 @@ pub fn handler(request: *AuthenticatedRequest, writer: anytype) ![]const u8 {
|
||||
}
|
||||
allocator.free(request_params.table_info.attribute_definitions);
|
||||
}
|
||||
var db = try dbForAccount(allocator, account_id);
|
||||
var db = try Account.dbForAccount(allocator, account_id);
|
||||
const account = try Account.accountForId(allocator, account_id); // This will get us the encryption key needed
|
||||
defer account.deinit();
|
||||
// TODO: better to do all encryption when request params are parsed?
|
||||
@ -243,46 +214,6 @@ fn insertIntoDm(
|
||||
@as(usize, 0),
|
||||
});
|
||||
}
|
||||
/// Gets the database for this account. If under test, a memory database is used
|
||||
/// instead. Will initialize the database with appropriate metadata tables
|
||||
fn dbForAccount(allocator: std.mem.Allocator, account_id: []const u8) !sqlite.Db {
|
||||
// TODO: Need to move this function somewhere central
|
||||
// TODO: Need configuration for what directory to use
|
||||
// TODO: Should this be a pool, and if so, how would we know when to close?
|
||||
const file_without_path = try std.fmt.allocPrint(allocator, "ddb-{s}.sqlite3", .{account_id});
|
||||
defer allocator.free(file_without_path);
|
||||
const db_file_name = try std.fs.path.joinZ(allocator, &[_][]const u8{ data_dir, file_without_path });
|
||||
defer allocator.free(db_file_name);
|
||||
const mode = if (@import("builtin").is_test) sqlite.Db.Mode.Memory else sqlite.Db.Mode{ .File = db_file_name };
|
||||
const new = mode == .Memory or (std.fs.cwd().statFile(file_without_path) catch null == null);
|
||||
var db = try sqlite.Db.init(.{
|
||||
.mode = mode,
|
||||
.open_flags = .{
|
||||
.write = true,
|
||||
.create = new,
|
||||
},
|
||||
.threading_mode = .MultiThread,
|
||||
});
|
||||
|
||||
// DDB minimum table name length is 3. DDB local creates this table with metadata
|
||||
// This of course is only if the database is first run
|
||||
if (new)
|
||||
try db.exec(
|
||||
\\CREATE TABLE dm (
|
||||
\\ TableName TEXT,
|
||||
\\ CreationDateTime INTEGER,
|
||||
\\ LastDecreaseDate INTEGER,
|
||||
\\ LastIncreaseDate INTEGER,
|
||||
\\ NumberOfDecreasesToday INTEGER,
|
||||
\\ ReadCapacityUnits INTEGER,
|
||||
\\ WriteCapacityUnits INTEGER,
|
||||
\\ TableInfo BLOB,
|
||||
\\ BillingMode INTEGER DEFAULT 0,
|
||||
\\ PayPerRequestDateTime INTEGER DEFAULT 0,
|
||||
\\ PRIMARY KEY(TableName))
|
||||
, .{}, .{});
|
||||
return db;
|
||||
}
|
||||
|
||||
fn parseRequest(
|
||||
request: *AuthenticatedRequest,
|
||||
@ -499,9 +430,9 @@ fn parseRequest(
|
||||
return request_params;
|
||||
}
|
||||
|
||||
fn parseAttributeDefinitions(request: *AuthenticatedRequest, definitions: []std.json.Value, writer: anytype) ![]*AttributeDefinition {
|
||||
fn parseAttributeDefinitions(request: *AuthenticatedRequest, definitions: []std.json.Value, writer: anytype) ![]*ddb_types.AttributeDefinition {
|
||||
const allocator = request.allocator;
|
||||
var rc = try allocator.alloc(*AttributeDefinition, definitions.len);
|
||||
var rc = try allocator.alloc(*ddb_types.AttributeDefinition, definitions.len);
|
||||
errdefer allocator.free(rc);
|
||||
// "AttributeDefinitions": [
|
||||
// {
|
||||
@ -533,7 +464,7 @@ fn parseAttributeDefinitions(request: *AuthenticatedRequest, definitions: []std.
|
||||
"Attribute definitions array can only consist of objects with AttributeName and AttributeType strings",
|
||||
);
|
||||
const type_string = attribute_type.?.string;
|
||||
const type_enum = std.meta.stringToEnum(AttributeTypeDescriptor, type_string);
|
||||
const type_enum = std.meta.stringToEnum(ddb_types.AttributeTypeDescriptor, type_string);
|
||||
if (type_enum == null)
|
||||
try returnException(
|
||||
request,
|
||||
@ -544,35 +475,13 @@ fn parseAttributeDefinitions(request: *AuthenticatedRequest, definitions: []std.
|
||||
); // TODO: This is kind of a lousy error message
|
||||
// TODO: This can leak memory if a later validation error occurs.
|
||||
// we are de-facto passed an arena here, but we shouldn't assume that
|
||||
var definition = try allocator.create(AttributeDefinition);
|
||||
var definition = try allocator.create(ddb_types.AttributeDefinition);
|
||||
definition.name = try allocator.dupe(u8, name.?.string);
|
||||
definition.type = type_enum.?;
|
||||
rc[i] = definition;
|
||||
}
|
||||
return rc;
|
||||
}
|
||||
fn returnException(
|
||||
request: *AuthenticatedRequest,
|
||||
status: std.http.Status,
|
||||
err: anyerror,
|
||||
writer: anytype,
|
||||
message: []const u8,
|
||||
) !void {
|
||||
switch (request.output_format) {
|
||||
.json => try writer.print(
|
||||
\\{{"__type":"{s}","message":"{s}"}}
|
||||
,
|
||||
.{ @errorName(err), message },
|
||||
),
|
||||
|
||||
.text => try writer.print(
|
||||
"{s}: {s}\n",
|
||||
.{ @errorName(err), message },
|
||||
),
|
||||
}
|
||||
request.status = status;
|
||||
return err;
|
||||
}
|
||||
// Full request syntax:
|
||||
//
|
||||
// {
|
||||
|
30
src/ddb_types.zig
Normal file
30
src/ddb_types.zig
Normal file
@ -0,0 +1,30 @@
|
||||
pub const AttributeTypeDescriptor = enum(u4) {
|
||||
S = 0,
|
||||
N = 1,
|
||||
B = 2,
|
||||
BOOL = 3,
|
||||
NULL = 4,
|
||||
M = 5,
|
||||
L = 6,
|
||||
SS = 7,
|
||||
NS = 8,
|
||||
BS = 9,
|
||||
};
|
||||
|
||||
pub const AttributeTypeName = enum(u4) {
|
||||
string = 0,
|
||||
number = 1,
|
||||
binary = 2,
|
||||
boolean = 3,
|
||||
null = 4,
|
||||
map = 5,
|
||||
list = 6,
|
||||
string_set = 7,
|
||||
number_set = 8,
|
||||
binary_set = 9,
|
||||
};
|
||||
|
||||
pub const AttributeDefinition = struct {
|
||||
name: []const u8,
|
||||
type: AttributeTypeDescriptor,
|
||||
};
|
29
src/main.zig
29
src/main.zig
@ -58,6 +58,10 @@ pub fn handler(allocator: std.mem.Allocator, event_data: []const u8, context: un
|
||||
const writer = context.writer();
|
||||
if (std.ascii.eqlIgnoreCase("CreateTable", operation))
|
||||
return executeOperation(&authenticated_request, context, writer, @import("createtable.zig").handler);
|
||||
if (std.ascii.eqlIgnoreCase("BatchWriteItem", operation))
|
||||
return executeOperation(&authenticated_request, context, writer, @import("batchwriteitem.zig").handler);
|
||||
if (std.ascii.eqlIgnoreCase("BatchGetItem", operation))
|
||||
return executeOperation(&authenticated_request, context, writer, @import("batchgetitem.zig").handler);
|
||||
|
||||
try writer.print("Operation '{s}' unsupported\n", .{operation});
|
||||
context.status = .bad_request;
|
||||
@ -140,6 +144,29 @@ fn accountId(allocator: std.mem.Allocator, headers: std.http.Headers) ![]const u
|
||||
return try iamAccountId(allocator);
|
||||
}
|
||||
|
||||
pub fn returnException(
|
||||
request: *AuthenticatedRequest,
|
||||
status: std.http.Status,
|
||||
err: anyerror,
|
||||
writer: anytype,
|
||||
message: []const u8,
|
||||
) !void {
|
||||
switch (request.output_format) {
|
||||
.json => try writer.print(
|
||||
\\{{"__type":"{s}","message":"{s}"}}
|
||||
,
|
||||
.{ @errorName(err), message },
|
||||
),
|
||||
|
||||
.text => try writer.print(
|
||||
"{s}: {s}\n",
|
||||
.{ @errorName(err), message },
|
||||
),
|
||||
}
|
||||
request.status = status;
|
||||
return err;
|
||||
}
|
||||
|
||||
// These never need to be freed because we will need them throughout the program
|
||||
var iam_account_id: ?[]const u8 = null;
|
||||
var iam_access_key: ?[]const u8 = null;
|
||||
@ -167,4 +194,6 @@ fn getVariable(allocator: std.mem.Allocator, global: *?[]const u8, env_var_name:
|
||||
|
||||
test {
|
||||
std.testing.refAllDecls(@import("createtable.zig"));
|
||||
std.testing.refAllDecls(@import("batchwriteitem.zig"));
|
||||
std.testing.refAllDecls(@import("batchgetitem.zig"));
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user