lib/grizzly/zwave/commands/crc_16_encap.ex

defmodule Grizzly.ZWave.Commands.CRC16Encap do
  @moduledoc """
  The CRC-16 Encapsulation Command is used to encapsulate a command with an additional checksum to
  ensure integrity of the payload.

  Params:

    * `:command` - the encapsulated Command

  """

  @behaviour Grizzly.ZWave.Command

  alias Grizzly.ZWave.{Command, DecodeError, CRC, Decoder}
  alias Grizzly.ZWave.CommandClasses.CRC16Encap

  @type param :: {:command, Command.t()}

  @impl true
  @spec new([param()]) :: {:ok, Command.t()}
  def new(params) do
    command = %Command{
      name: :crc_16_encap,
      command_byte: 0x01,
      command_class: CRC16Encap,
      params: params,
      impl: __MODULE__
    }

    {:ok, command}
  end

  @impl true
  @spec encode_params(Command.t()) :: binary()
  def encode_params(command) do
    command = Command.param!(command, :command)
    command_class_byte = apply(command.command_class, :byte, [])
    command_byte = command.command_byte
    encoded_params = apply(command.impl, :encode_params, [command])
    command_binary = <<command_class_byte, command_byte>> <> encoded_params
    checksum = CRC.crc16_aug_ccitt(command_binary)
    command_binary <> <<checksum::size(2)-integer-unsigned-unit(8)>>
  end

  @impl true
  @spec decode_params(binary()) :: {:ok, [param()]} | {:error, DecodeError.t()}
  def decode_params(<<command_class_byte, command_byte, data_and_checksum::binary>> = binary) do
    with {:ok, encoded_params, checksum} <- extract_params_and_checksum(data_and_checksum),
         command_binary = <<command_class_byte, command_byte>> <> encoded_params,
         ^checksum <- CRC.crc16_aug_ccitt(command_binary),
         {:ok, command} <- Decoder.from_binary(command_binary) do
      {:ok, [command: command]}
    else
      _other ->
        {:error, %DecodeError{value: binary, param: :command, command: :crc_16_encap}}
    end
  end

  defp extract_params_and_checksum(data_and_checksum) do
    case :erlang.binary_to_list(data_and_checksum) |> Enum.reverse() do
      [checksum_2, checksum_1 | reversed_param_bytes] ->
        checksum_binary = <<checksum_1, checksum_2>>
        <<checksum::size(2)-integer-unsigned-unit(8)>> = checksum_binary
        encoded_params = :erlang.list_to_binary(Enum.reverse(reversed_param_bytes))
        {:ok, encoded_params, checksum}

      _other ->
        {:error, :invalid}
    end
  end
end