mirror of
https://github.com/ziglang/zig.git
synced 2025-12-06 22:33:08 +00:00
140 lines
5.3 KiB
Zig
140 lines
5.3 KiB
Zig
//! JSON parsing and stringification conforming to RFC 8259. https://datatracker.ietf.org/doc/html/rfc8259
|
|
//!
|
|
//! The low-level `Scanner` API produces `Token`s from an input slice or successive slices of inputs,
|
|
//! The `Reader` API connects a `std.io.GenericReader` to a `Scanner`.
|
|
//!
|
|
//! The high-level `parseFromSlice` and `parseFromTokenSource` deserialize a JSON document into a Zig type.
|
|
//! Parse into a dynamically-typed `Value` to load any JSON value for runtime inspection.
|
|
//!
|
|
//! The low-level `writeStream` emits syntax-conformant JSON tokens to a `std.io.GenericWriter`.
|
|
//! The high-level `stringify` serializes a Zig or `Value` type into JSON.
|
|
|
|
const builtin = @import("builtin");
|
|
const std = @import("std");
|
|
const testing = std.testing;
|
|
|
|
test Scanner {
|
|
var scanner = Scanner.initCompleteInput(testing.allocator, "{\"foo\": 123}\n");
|
|
defer scanner.deinit();
|
|
try testing.expectEqual(Token.object_begin, try scanner.next());
|
|
try testing.expectEqualSlices(u8, "foo", (try scanner.next()).string);
|
|
try testing.expectEqualSlices(u8, "123", (try scanner.next()).number);
|
|
try testing.expectEqual(Token.object_end, try scanner.next());
|
|
try testing.expectEqual(Token.end_of_document, try scanner.next());
|
|
}
|
|
|
|
test parseFromSlice {
|
|
var parsed_str = try parseFromSlice([]const u8, testing.allocator, "\"a\\u0020b\"", .{});
|
|
defer parsed_str.deinit();
|
|
try testing.expectEqualSlices(u8, "a b", parsed_str.value);
|
|
|
|
const T = struct { a: i32 = -1, b: [2]u8 };
|
|
var parsed_struct = try parseFromSlice(T, testing.allocator, "{\"b\":\"xy\"}", .{});
|
|
defer parsed_struct.deinit();
|
|
try testing.expectEqual(@as(i32, -1), parsed_struct.value.a); // default value
|
|
try testing.expectEqualSlices(u8, "xy", parsed_struct.value.b[0..]);
|
|
}
|
|
|
|
test Value {
|
|
var parsed = try parseFromSlice(Value, testing.allocator, "{\"anything\": \"goes\"}", .{});
|
|
defer parsed.deinit();
|
|
try testing.expectEqualSlices(u8, "goes", parsed.value.object.get("anything").?.string);
|
|
}
|
|
|
|
test Stringify {
|
|
var out: std.io.Writer.Allocating = .init(testing.allocator);
|
|
var write_stream: Stringify = .{
|
|
.writer = &out.writer,
|
|
.options = .{ .whitespace = .indent_2 },
|
|
};
|
|
defer out.deinit();
|
|
try write_stream.beginObject();
|
|
try write_stream.objectField("foo");
|
|
try write_stream.write(123);
|
|
try write_stream.endObject();
|
|
const expected =
|
|
\\{
|
|
\\ "foo": 123
|
|
\\}
|
|
;
|
|
try testing.expectEqualSlices(u8, expected, out.getWritten());
|
|
}
|
|
|
|
pub const ObjectMap = @import("json/dynamic.zig").ObjectMap;
|
|
pub const Array = @import("json/dynamic.zig").Array;
|
|
pub const Value = @import("json/dynamic.zig").Value;
|
|
|
|
pub const ArrayHashMap = @import("json/hashmap.zig").ArrayHashMap;
|
|
|
|
pub const Scanner = @import("json/Scanner.zig");
|
|
pub const validate = Scanner.validate;
|
|
pub const Error = Scanner.Error;
|
|
pub const default_buffer_size = Scanner.default_buffer_size;
|
|
pub const Token = Scanner.Token;
|
|
pub const TokenType = Scanner.TokenType;
|
|
pub const Diagnostics = Scanner.Diagnostics;
|
|
pub const AllocWhen = Scanner.AllocWhen;
|
|
pub const default_max_value_len = Scanner.default_max_value_len;
|
|
pub const Reader = Scanner.Reader;
|
|
pub const isNumberFormattedLikeAnInteger = Scanner.isNumberFormattedLikeAnInteger;
|
|
|
|
pub const ParseOptions = @import("json/static.zig").ParseOptions;
|
|
pub const Parsed = @import("json/static.zig").Parsed;
|
|
pub const parseFromSlice = @import("json/static.zig").parseFromSlice;
|
|
pub const parseFromSliceLeaky = @import("json/static.zig").parseFromSliceLeaky;
|
|
pub const parseFromTokenSource = @import("json/static.zig").parseFromTokenSource;
|
|
pub const parseFromTokenSourceLeaky = @import("json/static.zig").parseFromTokenSourceLeaky;
|
|
pub const innerParse = @import("json/static.zig").innerParse;
|
|
pub const parseFromValue = @import("json/static.zig").parseFromValue;
|
|
pub const parseFromValueLeaky = @import("json/static.zig").parseFromValueLeaky;
|
|
pub const innerParseFromValue = @import("json/static.zig").innerParseFromValue;
|
|
pub const ParseError = @import("json/static.zig").ParseError;
|
|
pub const ParseFromValueError = @import("json/static.zig").ParseFromValueError;
|
|
|
|
pub const Stringify = @import("json/Stringify.zig");
|
|
|
|
/// Returns a formatter that formats the given value using stringify.
|
|
pub fn fmt(value: anytype, options: Stringify.Options) Formatter(@TypeOf(value)) {
|
|
return Formatter(@TypeOf(value)){ .value = value, .options = options };
|
|
}
|
|
|
|
test fmt {
|
|
const expectFmt = std.testing.expectFmt;
|
|
try expectFmt("123", "{f}", .{fmt(@as(u32, 123), .{})});
|
|
try expectFmt(
|
|
\\{"num":927,"msg":"hello","sub":{"mybool":true}}
|
|
, "{f}", .{fmt(struct {
|
|
num: u32,
|
|
msg: []const u8,
|
|
sub: struct {
|
|
mybool: bool,
|
|
},
|
|
}{
|
|
.num = 927,
|
|
.msg = "hello",
|
|
.sub = .{ .mybool = true },
|
|
}, .{})});
|
|
}
|
|
|
|
/// Formats the given value using stringify.
|
|
pub fn Formatter(comptime T: type) type {
|
|
return struct {
|
|
value: T,
|
|
options: Stringify.Options,
|
|
|
|
pub fn format(self: @This(), writer: *std.Io.Writer) std.Io.Writer.Error!void {
|
|
try Stringify.value(self.value, self.options, writer);
|
|
}
|
|
};
|
|
}
|
|
|
|
test {
|
|
_ = @import("json/test.zig");
|
|
_ = Scanner;
|
|
_ = @import("json/dynamic.zig");
|
|
_ = @import("json/hashmap.zig");
|
|
_ = @import("json/static.zig");
|
|
_ = Stringify;
|
|
_ = @import("json/JSONTestSuite_test.zig");
|
|
}
|