Function format [src]

Renders fmt string with args, calling writer with slices of bytes. If writer returns an error, the error is returned from format and writer is not called again. The format string must be comptime-known and may contain placeholders following this format: {[argument][specifier]:[fill][alignment][width].[precision]} Above, each word including its surrounding [ and ] is a parameter which you have to replace with something: argument is either the numeric index or the field name of the argument that should be inserted when using a field name, you are required to enclose the field name (an identifier) in square brackets, e.g. {[score]...} as opposed to the numeric index form which can be written e.g. {2...} specifier is a type-dependent formatting option that determines how a type should formatted (see below) fill is a single unicode codepoint which is used to pad the formatted text alignment is one of the three bytes '<', '^', or '>' to make the text left-, center-, or right-aligned, respectively width is the total width of the field in unicode codepoints precision specifies how many decimals a formatted number should have Note that most of the parameters are optional and may be omitted. Also you can leave out separators like : and . when all parameters after the separator are omitted. Only exception is the fill parameter. If a non-zero fill character is required at the same time as width is specified, one has to specify alignment as well, as otherwise the digit following : is interpreted as width, not fill. The specifier has several options for types: x and X: output numeric value in hexadecimal notation s: for pointer-to-many and C pointers of u8, print as a C-string using zero-termination for slices of u8, print the entire slice as a string without zero-termination e: output floating point value in scientific notation d: output numeric value in decimal notation b: output integer value in binary notation o: output integer value in octal notation c: output integer as an ASCII character. Integer type must have 8 bits at max. u: output integer as an UTF-8 sequence. Integer type must have 21 bits at max. ?: output optional value as either the unwrapped value, or null; may be followed by a format specifier for the underlying value. !: output error union value as either the unwrapped value, or the formatted error value; may be followed by a format specifier for the underlying value. *: output the address of the value instead of the value itself. any: output a value of any type using its default format. If a formatted user type contains a function of the type pub fn format(value: ?, comptime fmt: []const u8, options: std.fmt.FormatOptions, writer: anytype) !void with ? being the type formatted, this function will be called instead of the default implementation. This allows user types to be formatted in a logical manner instead of dumping all fields of the type. A user type may be a struct, vector, union or enum type. To print literal curly braces, escape them by writing them twice, e.g. {{ or }}.

Prototype

pub fn format( writer: anytype, comptime fmt: []const u8, args: anytype, ) !void

Parameters

fmt: []const u8

Source

pub fn format( writer: anytype, comptime fmt: []const u8, args: anytype, ) !void { const ArgsType = @TypeOf(args); const args_type_info = @typeInfo(ArgsType); if (args_type_info != .@"struct") { @compileError("expected tuple or struct argument, found " ++ @typeName(ArgsType)); } const fields_info = args_type_info.@"struct".fields; if (fields_info.len > max_format_args) { @compileError("32 arguments max are supported per format call"); } @setEvalBranchQuota(2000000); comptime var arg_state: ArgState = .{ .args_len = fields_info.len }; comptime var i = 0; comptime var literal: []const u8 = ""; inline while (true) { const start_index = i; inline while (i < fmt.len) : (i += 1) { switch (fmt[i]) { '{', '}' => break, else => {}, } } comptime var end_index = i; comptime var unescape_brace = false; // Handle {{ and }}, those are un-escaped as single braces if (i + 1 < fmt.len and fmt[i + 1] == fmt[i]) { unescape_brace = true; // Make the first brace part of the literal... end_index += 1; // ...and skip both i += 2; } literal = literal ++ fmt[start_index..end_index]; // We've already skipped the other brace, restart the loop if (unescape_brace) continue; // Write out the literal if (literal.len != 0) { try writer.writeAll(literal); literal = ""; } if (i >= fmt.len) break; if (fmt[i] == '}') { @compileError("missing opening {"); } // Get past the { comptime assert(fmt[i] == '{'); i += 1; const fmt_begin = i; // Find the closing brace inline while (i < fmt.len and fmt[i] != '}') : (i += 1) {} const fmt_end = i; if (i >= fmt.len) { @compileError("missing closing }"); } // Get past the } comptime assert(fmt[i] == '}'); i += 1; const placeholder = comptime Placeholder.parse(fmt[fmt_begin..fmt_end].*); const arg_pos = comptime switch (placeholder.arg) { .none => null, .number => |pos| pos, .named => |arg_name| meta.fieldIndex(ArgsType, arg_name) orelse @compileError("no argument with name '" ++ arg_name ++ "'"), }; const width = switch (placeholder.width) { .none => null, .number => |v| v, .named => |arg_name| blk: { const arg_i = comptime meta.fieldIndex(ArgsType, arg_name) orelse @compileError("no argument with name '" ++ arg_name ++ "'"); _ = comptime arg_state.nextArg(arg_i) orelse @compileError("too few arguments"); break :blk @field(args, arg_name); }, }; const precision = switch (placeholder.precision) { .none => null, .number => |v| v, .named => |arg_name| blk: { const arg_i = comptime meta.fieldIndex(ArgsType, arg_name) orelse @compileError("no argument with name '" ++ arg_name ++ "'"); _ = comptime arg_state.nextArg(arg_i) orelse @compileError("too few arguments"); break :blk @field(args, arg_name); }, }; const arg_to_print = comptime arg_state.nextArg(arg_pos) orelse @compileError("too few arguments"); try formatType( @field(args, fields_info[arg_to_print].name), placeholder.specifier_arg, FormatOptions{ .fill = placeholder.fill, .alignment = placeholder.alignment, .width = width, .precision = precision, }, writer, std.options.fmt_max_depth, ); } if (comptime arg_state.hasUnusedArgs()) { const missing_count = arg_state.args_len - @popCount(arg_state.used_args); switch (missing_count) { 0 => unreachable, 1 => @compileError("unused argument in '" ++ fmt ++ "'"), else => @compileError(comptimePrint("{d}", .{missing_count}) ++ " unused arguments in '" ++ fmt ++ "'"), } } }