Add encoder

This commit is contained in:
Omar Roth 2019-09-20 15:39:47 -04:00
parent d0c297bc7e
commit 6a4f04d0a3
No known key found for this signature in database
GPG key ID: B8254FB7EC3D37F2

View file

@ -38,6 +38,22 @@ struct VarLong
result result
end end
def self.to_io(io : IO, value : Int64)
io.write_byte 0x00 if value == 0x00
value = value.to_u64
while value != 0
byte = (value & 0x7f).to_u8
value >>= 7
if value != 0
byte |= 0x80
end
io.write_byte byte
end
end
end end
struct ProtoBuf::Any struct ProtoBuf::Any
@ -48,6 +64,18 @@ struct ProtoBuf::Any
Bit32 = 5 Bit32 = 5
end end
TAG_MAP = {
"varint" => 0,
"float32" => 5,
"int32" => 5,
"float64" => 1,
"int64" => 1,
"string" => 2,
"embedded" => 2,
"base64" => 2,
"bytes" => 2,
}
alias Type = Int64 | alias Type = Int64 |
Float64 | Float64 |
Array(UInt8) | Array(UInt8) |
@ -76,6 +104,7 @@ struct ProtoBuf::Any
case type case type
when Tag::VarInt when Tag::VarInt
value = io.read_bytes(VarLong) value = io.read_bytes(VarLong)
key = "#{field}:#{index}:varint"
when Tag::Bit32 when Tag::Bit32
value = io.read_bytes(Int32) value = io.read_bytes(Int32)
bytes = IO::Memory.new bytes = IO::Memory.new
@ -84,8 +113,10 @@ struct ProtoBuf::Any
begin begin
value = bytes.read_bytes(Float32, format: IO::ByteFormat::LittleEndian).to_f64 value = bytes.read_bytes(Float32, format: IO::ByteFormat::LittleEndian).to_f64
key = "#{field}:#{index}:float32"
rescue ex rescue ex
value = value.to_i64 value = value.to_i64
key = "#{field}:#{index}:int32"
end end
when Tag::Bit64 when Tag::Bit64
value = io.read_bytes(Int64) value = io.read_bytes(Int64)
@ -95,11 +126,13 @@ struct ProtoBuf::Any
begin begin
value = bytes.read_bytes(Float64, format: IO::ByteFormat::LittleEndian) value = bytes.read_bytes(Float64, format: IO::ByteFormat::LittleEndian)
key = "#{field}:#{index}:float64"
rescue ex rescue ex
key = "#{field}:#{index}:int64"
end end
when Tag::LengthDelimited when Tag::LengthDelimited
size = io.read_bytes(VarLong) size = io.read_bytes(VarLong)
raise "Invalid size" if size > 2**20 raise "Invalid size" if size > 2**22
bytes = Bytes.new(size) bytes = Bytes.new(size)
io.read_fully(bytes) io.read_fully(bytes)
@ -107,26 +140,31 @@ struct ProtoBuf::Any
value = String.new(bytes) value = String.new(bytes)
if value.empty? if value.empty?
value = "" value = ""
key = "#{field}:#{index}:string"
elsif value.valid_encoding? && !value.codepoints.any? { |codepoint| elsif value.valid_encoding? && !value.codepoints.any? { |codepoint|
(0x00..0x1f).includes?(codepoint) && (0x00..0x1f).includes?(codepoint) &&
!{0x09, 0x0a, 0x0d}.includes?(codepoint) !{0x09, 0x0a, 0x0d}.includes?(codepoint)
} }
begin begin
value = from_io(IO::Memory.new(Base64.decode(URI.unescape(URI.unescape(value))))).raw value = from_io(IO::Memory.new(Base64.decode(URI.unescape(URI.unescape(value))))).raw
key = "#{field}:#{index}:base64"
rescue ex rescue ex
key = "#{field}:#{index}:string"
end end
else else
begin begin
value = from_io(IO::Memory.new(bytes)).raw value = from_io(IO::Memory.new(bytes)).raw
key = "#{field}:#{index}:embedded"
rescue ex rescue ex
value = bytes.to_a value = bytes.to_a
key = "#{field}:#{index}:bytes"
end end
end end
else else
raise "Invalid type #{type}" raise "Invalid type #{type}"
end end
item["#{field}:#{index}"] = value.as(Type) item[key] = value.as(Type)
index += 1 index += 1
end end
rescue ex rescue ex
@ -152,30 +190,79 @@ struct ProtoBuf::Any
def to_json(json) def to_json(json)
raw.to_json(json) raw.to_json(json)
end end
def self.from_json(json : JSON::Any, io : IO, format = IO::ByteFormat::NetworkEndian)
case object = json.raw
when Hash
object.each do |key, value|
parts = key.split(":")
field = parts[0].to_i64
type = parts[-1]
header = (field << 3) | TAG_MAP[type]
VarLong.to_io(io, header)
case type
when "varint"
VarLong.to_io(io, value.as_i64)
when "int32"
value.as_i64.to_i32.to_io(io, IO::ByteFormat::LittleEndian)
when "float32"
value.as_f32.to_f32.to_io(io, IO::ByteFormat::LittleEndian)
when "int64"
value.as_i64.to_io(io, IO::ByteFormat::LittleEndian)
when "float64"
value.as_f32.to_f64.to_io(io, IO::ByteFormat::LittleEndian)
when "string"
VarLong.to_io(io, value.as_s.size.to_i64)
value.as_s.to_s(io)
when "base64"
buffer = IO::Memory.new
from_json(value, buffer)
buffer.rewind
buffer = Base64.urlsafe_encode(buffer, padding: false)
VarLong.to_io(io, buffer.size.to_i64)
buffer.to_s(io)
when "embedded"
buffer = IO::Memory.new
from_json(value, buffer)
buffer.rewind
VarLong.to_io(io, buffer.size.to_i64)
IO.copy(buffer, io)
when "bytes"
VarLong.to_io(io, value.size.to_i64)
value.as_a.each { |byte| io.write_byte byte.as_i.to_u8 }
end
end
else
raise "Invalid value #{json}"
end
end
end end
enum InputType enum IOType
Base64 Base64
Hex Hex
Raw Raw
end
enum OutputType
Json Json
JsonPretty JsonPretty
end end
input_type = InputType::Raw input_type = nil
output_type = OutputType::Json output_type = nil
flags = [] of String flags = [] of String
OptionParser.parse! do |parser| OptionParser.parse! do |parser|
parser.banner = <<-'END_USAGE' parser.banner = <<-'END_USAGE'
Usage: protodec [arguments] Usage: protodec [arguments]
Command-line decoder for arbitrary protobuf data. Reads from standard input. Command-line encoder and decoder for arbitrary protobuf data. Reads from standard input.
END_USAGE END_USAGE
parser.on("-d", "--decode", "STDIN is Base64-encoded") { flags << "d" } parser.on("-e", "--encode", "Encode input") { flags << "e" }
parser.on("-d", "--decode", "Decode input (default)") { flags << "d" }
parser.on("-b", "--base64", "STDIN is Base64-encoded") { flags << "b" }
parser.on("-x", "--hex", "STDIN is space-separated hexstring") { flags << "x" } parser.on("-x", "--hex", "STDIN is space-separated hexstring") { flags << "x" }
parser.on("-r", "--raw", "STDIN is raw binary data (default)") { flags << "r" } parser.on("-r", "--raw", "STDIN is raw binary data (default)") { flags << "r" }
parser.on("-p", "--pretty", "Pretty print output") { flags << "p" } parser.on("-p", "--pretty", "Pretty print output") { flags << "p" }
@ -188,34 +275,57 @@ end
flags.each do |flag| flags.each do |flag|
case flag case flag
when "d" when "b"
input_type = InputType::Base64 input_type = IOType::Base64
when "x" when "x"
input_type = InputType::Hex input_type = IOType::Hex
when "r" when "r"
input_type = InputType::Raw input_type = IOType::Raw
when "p" when "p"
output_type = OutputType::JsonPretty output_type = IOType::JsonPretty
when "e", "d"
else else
STDERR.puts "ERROR: #{flag} is not a valid option." STDERR.puts "ERROR: #{flag} is not a valid option."
exit(1) exit(1)
end end
end end
input = STDIN.gets_to_end if flags.includes? "e"
case input_type tmp = output_type
when InputType::Base64 output_type = input_type
input = Base64.decode(URI.unescape(URI.unescape(input.strip))) input_type = tmp
when InputType::Hex
array = input.strip.split(/[- ,]+/).map &.to_i(16).to_u8 input_type ||= IOType::Json
input = Slice.new(array.size) { |i| array[i] } output_type ||= IOType::Base64
when InputType::Raw else
input_type ||= IOType::Raw
output_type ||= IOType::Json
end end
output = ProtoBuf::Any.parse(IO::Memory.new(input)) case input_type
case output_type when IOType::Base64
when OutputType::Json output = ProtoBuf::Any.parse(IO::Memory.new(Base64.decode(URI.unescape(URI.unescape(STDIN.gets_to_end.strip)))))
STDOUT.puts output.to_json when IOType::Hex
when OutputType::JsonPretty array = STDIN.gets_to_end.strip.split(/[- ,]+/).map &.to_i(16).to_u8
STDOUT.puts output.to_pretty_json output = ProtoBuf::Any.parse(IO::Memory.new(Slice.new(array.size) { |i| array[i] }))
when IOType::Raw
output = ProtoBuf::Any.parse(IO::Memory.new(STDIN.gets_to_end))
when IOType::Json, IOType::JsonPretty
output = IO::Memory.new
ProtoBuf::Any.from_json(JSON.parse(STDIN), output)
else
output = ""
end
case output_type
when IOType::Base64
STDOUT.puts Base64.urlsafe_encode(output.as(IO))
when IOType::Hex
STDOUT.puts (output.as(IO).to_slice.map &.to_s(16).rjust(2, '0').upcase).join("-")
when IOType::Raw
STDOUT.write output.as(IO).to_slice
when IOType::Json
STDOUT.puts output.as(ProtoBuf::Any).to_json
when IOType::JsonPretty
STDOUT.puts output.as(ProtoBuf::Any).to_pretty_json
end end