Skip to content

Commit

Permalink
Implement encoding and decoding of properties
Browse files Browse the repository at this point in the history
This should help implementing a considerable amount of the MQTT 5
protocol.
  • Loading branch information
gausby committed Sep 12, 2018
1 parent 2a9ed23 commit c305d9a
Show file tree
Hide file tree
Showing 4 changed files with 360 additions and 0 deletions.
15 changes: 15 additions & 0 deletions lib/tortoise/package.ex
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,11 @@ defmodule Tortoise.Package do
[length_prefix, data]
end

@doc false
def variable_length(n) do
remaining_length(n)
end

@doc false
def variable_length_encode(data) when is_list(data) do
length_prefix = data |> IO.iodata_length() |> remaining_length()
Expand All @@ -40,4 +45,14 @@ defmodule Tortoise.Package do
defp remaining_length(n) do
[<<1::1, rem(n, @highbit)::7>>] ++ remaining_length(div(n, @highbit))
end

@doc false
def drop_length_prefix(payload) do
case payload do
<<0::1, _::7, r::binary>> -> r
<<1::1, _::7, 0::1, _::7, r::binary>> -> r
<<1::1, _::7, 1::1, _::7, 0::1, _::7, r::binary>> -> r
<<1::1, _::7, 1::1, _::7, 1::1, _::7, 0::1, _::7, r::binary>> -> r
end
end
end
257 changes: 257 additions & 0 deletions lib/tortoise/package/properties.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,257 @@
defmodule Tortoise.Package.Properties do
@moduledoc false

alias Tortoise.Package

import Tortoise.Package, only: [variable_length: 1, length_encode: 1]

def encode(data) when is_list(data) do
data
|> Enum.map(&encode_property/1)
|> Package.variable_length_encode()
end

# User properties are special; we will allow them to be encoded as
# binaries to make the interface a bit cleaner to the end user
defp encode_property({key, value}) when is_binary(key) do
[0x26, length_encode(key), length_encode(value)]
end

defp encode_property({key, value}) do
case key do
:payload_format_indicator ->
[0x01, <<value::8>>]

:message_expiry_interval ->
[0x02, <<value::integer-size(32)>>]

:content_type ->
[0x03, length_encode(value)]

:response_topic ->
[0x08, length_encode(value)]

:correlation_data ->
[0x09, length_encode(value)]

:subscription_identifier when is_integer(value) ->
[0x0B, variable_length(value)]

:session_expiry_interval ->
[0x11, <<value::integer-size(32)>>]

:assigned_client_identifier ->
[0x12, length_encode(value)]

:server_keep_alive ->
[0x13, <<value::integer-size(16)>>]

:authentication_method ->
[0x15, length_encode(value)]

:authentication_data when is_binary(value) ->
[0x16, length_encode(value)]

:request_problem_information ->
[0x17, <<value::8>>]

:will_delay_interval when is_integer(value) ->
[0x18, <<value::integer-size(32)>>]

:request_response_information ->
[0x19, <<value::8>>]

:response_information ->
[0x1A, length_encode(value)]

:server_reference ->
[0x1C, length_encode(value)]

:reason_string ->
[0x1F, length_encode(value)]

:receive_maximum ->
[0x21, <<value::integer-size(16)>>]

:topic_alias_maximum ->
[0x22, <<value::integer-size(16)>>]

:topic_alias ->
[0x23, <<value::integer-size(16)>>]

:maximum_qos ->
[0x24, <<value::8>>]

:retain_available ->
[0x25, <<value::8>>]

:maximum_packet_size ->
[0x27, <<value::integer-size(32)>>]

:wildcard_subscription_available ->
[0x28, <<value::8>>]

:subscription_identifier_available ->
[0x29, <<value::8>>]

:shared_subscription_available ->
[0x2A, <<value::8>>]
end
end

# ---
def decode(data) do
data
|> Package.drop_length_prefix()
|> do_decode()
end

defp do_decode(data) do
data
|> decode_property()
|> case do
{nil, <<>>} -> []
{decoded, rest} -> [decoded] ++ do_decode(rest)
end
end

defp decode_property(<<>>) do
{nil, <<>>}
end

defp decode_property(<<0x01, value::8, rest::binary>>) do
{{:payload_format_indicator, value}, rest}
end

defp decode_property(<<0x02, value::integer-size(32), rest::binary>>) do
{{:message_expiry_interval, value}, rest}
end

defp decode_property(<<0x03, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:content_type, value}, rest}
end

defp decode_property(<<0x08, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:response_topic, value}, rest}
end

defp decode_property(<<0x09, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:correlation_data, value}, rest}
end

defp decode_property(<<0x0B, rest::binary>>) do
case rest do
<<0::1, value::integer-size(7), rest::binary>> ->
{{:subscription_identifier, value}, rest}

<<1::1, a::7, 0::1, b::7, rest::binary>> ->
<<value::integer-size(14)>> = <<b::7, a::7>>
{{:subscription_identifier, value}, rest}

<<1::1, a::7, 1::1, b::7, 0::1, c::7, rest::binary>> ->
<<value::integer-size(21)>> = <<c::7, b::7, a::7>>
{{:subscription_identifier, value}, rest}

<<1::1, a::7, 1::1, b::7, 1::1, c::7, 0::1, d::7, rest::binary>> ->
<<value::integer-size(28)>> = <<d::7, c::7, b::7, a::7>>
{{:subscription_identifier, value}, rest}
end
end

defp decode_property(<<0x11, value::integer-size(32), rest::binary>>) do
{{:session_expiry_interval, value}, rest}
end

defp decode_property(<<0x12, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:assigned_client_identifier, value}, rest}
end

defp decode_property(<<0x13, value::integer-size(16), rest::binary>>) do
{{:server_keep_alive, value}, rest}
end

defp decode_property(<<0x15, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:authentication_method, value}, rest}
end

defp decode_property(<<0x16, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:authentication_data, value}, rest}
end

defp decode_property(<<0x17, value::8, rest::binary>>) do
{{:request_problem_information, value}, rest}
end

defp decode_property(<<0x18, value::integer-size(32), rest::binary>>) do
{{:will_delay_interval, value}, rest}
end

defp decode_property(<<0x19, value::8, rest::binary>>) do
{{:request_response_information, value}, rest}
end

defp decode_property(<<0x1A, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:response_information, value}, rest}
end

defp decode_property(<<0x1C, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:server_reference, value}, rest}
end

defp decode_property(<<0x1F, length::integer-size(16), rest::binary>>) do
<<value::binary-size(length), rest::binary>> = rest
{{:reason_string, value}, rest}
end

defp decode_property(<<0x21, value::integer-size(16), rest::binary>>) do
{{:receive_maximum, value}, rest}
end

defp decode_property(<<0x22, value::integer-size(16), rest::binary>>) do
{{:topic_alias_maximum, value}, rest}
end

defp decode_property(<<0x23, value::integer-size(16), rest::binary>>) do
{{:topic_alias, value}, rest}
end

defp decode_property(<<0x24, value::8, rest::binary>>) do
{{:maximum_qos, value}, rest}
end

defp decode_property(<<0x25, value::8, rest::binary>>) do
{{:retain_available, value}, rest}
end

defp decode_property(<<0x26, rest::binary>>) do
<<length::integer-size(16), rest::binary>> = rest
<<key::binary-size(length), rest::binary>> = rest
<<length::integer-size(16), rest::binary>> = rest
<<value::binary-size(length), rest::binary>> = rest
{{key, value}, rest}
end

defp decode_property(<<0x27, value::integer-size(32), rest::binary>>) do
{{:maximum_packet_size, value}, rest}
end

defp decode_property(<<0x28, value::8, rest::binary>>) do
{{:wildcard_subscription_available, value}, rest}
end

defp decode_property(<<0x29, value::8, rest::binary>>) do
{{:subscription_identifier_available, value}, rest}
end

defp decode_property(<<0x2A, value::8, rest::binary>>) do
{{:shared_subscription_available, value}, rest}
end
end
67 changes: 67 additions & 0 deletions test/test_helper.exs
Original file line number Diff line number Diff line change
Expand Up @@ -253,6 +253,73 @@ defmodule Tortoise.TestGenerators do
pubrec
end
end

def gen_properties() do
let properties <-
oneof([
:payload_format_indicator,
:message_expiry_interval,
:content_type,
:response_topic,
:correlation_data,
:subscription_identifier,
:session_expiry_interval,
:assigned_client_identifier,
:server_keep_alive,
:authentication_method,
:authentication_data,
:request_problem_information,
:will_delay_interval,
:request_response_information,
:response_information,
:server_reference,
:reason_string,
:receive_maximum,
:topic_alias_maximum,
:topic_alias,
:maximum_qos,
:retain_available,
:user_property,
:maximum_packet_size,
:wildcard_subscription_available,
:subscription_identifier_available,
:shared_subscription_available
]) do
list(10, lazy(do: gen_property_value(properties)))
end
end

def gen_property_value(type) do
case type do
:payload_format_indicator -> {type, oneof([0, 1])}
:message_expiry_interval -> {type, choose(0, 4_294_967_295)}
:content_type -> {type, utf8()}
:response_topic -> {type, gen_topic()}
:correlation_data -> {type, binary()}
:subscription_identifier -> {type, choose(1, 268_435_455)}
:session_expiry_interval -> {type, choose(1, 268_435_455)}
:assigned_client_identifier -> {type, utf8()}
:server_keep_alive -> {type, choose(0x0000, 0xFFFF)}
:authentication_method -> {type, utf8()}
:authentication_data -> {type, binary()}
:request_problem_information -> {type, oneof([0, 1])}
:will_delay_interval -> {type, choose(0, 4_294_967_295)}
:request_response_information -> {type, oneof([0, 1])}
:response_information -> {type, utf8()}
:server_reference -> {type, utf8()}
:reason_string -> {type, utf8()}
:receive_maximum -> {type, choose(0x0001, 0xFFFF)}
:topic_alias_maximum -> {type, choose(0x0000, 0xFFFF)}
:topic_alias -> {type, choose(0x0001, 0xFFFF)}
:maximum_qos -> {type, oneof([0, 1])}
:retain_available -> {type, oneof([0, 1])}
:user_property -> {utf8(), utf8()}
:maximum_packet_size -> {type, choose(1, 268_435_455)}
:wildcard_subscription_available -> {type, oneof([0, 1])}
:subscription_identifier_available -> {type, oneof([0, 1])}
:shared_subscription_available -> {type, oneof([0, 1])}
end
end
end

# make certs for tests using the SSL transport
Expand Down
21 changes: 21 additions & 0 deletions test/tortoise/package/properties_test.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
defmodule Tortoise.Package.PropertiesTest do
use ExUnit.Case
use EQC.ExUnit
doctest Tortoise.Package.Properties

alias Tortoise.Package.Properties

import Tortoise.TestGenerators, only: [gen_properties: 0]

property "encoding and decoding properties" do
forall properties <- gen_properties() do
ensure(
properties ==
properties
|> Properties.encode()
|> IO.iodata_to_binary()
|> Properties.decode()
)
end
end
end

0 comments on commit c305d9a

Please sign in to comment.