Base58在Elixir中返回不正确的值

时间:2018-12-19 15:34:02

标签: elixir bitcoin base58

我正在尝试将十六进制编码为base58。它适用于js库base-x

十六进制= 1777c7ba65e23151ec09125011dd25c28998c70230e7b89ca6

预期

base58 = AShDKgLSuCjGZr8Fs5SRLSYvmcSV7S4zwX

知道

base58 = cAvfov2bvPACeGktuSEtz6G526UBfCwpia1354fp5bYJwP2rhhnxqYkRcwRoDrmgqZaG

defmodule Base58 do
  @alphabet '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'

  def encode(data, hash \\ "")

  def encode(data, hash) when is_binary(data) do
    encode_zeros(data) <> encode(:binary.decode_unsigned(data), hash)
  end

  def encode(0, hash), do: hash

  def encode(data, hash) do
    character = <<Enum.at(@alphabet, rem(data, 58))>>
    encode(div(data, 58), character <> hash)
  end

  defp encode_zeros(data) do
    <<Enum.at(@alphabet, 0)>>
    |> String.duplicate(leading_zeros(data))
  end

  defp leading_zeros(data) do
    :binary.bin_to_list(data)
    |> Enum.find_index(&(&1 != 0))
  end
end

1 个答案:

答案 0 :(得分:3)

正在发生的事情是,输入应该是由那些十六进制数字表示的二进制数据块,但是您将包含十六进制数字的字符串传递给了Base58.encode。如果将输入作为整数传递(使用0x前缀将其标记为十六进制文字整数),则会得到预期的结果:

iex(2)> Base58.encode("1777c7ba65e23151ec09125011dd25c28998c70230e7b89ca6")
"cAvfov2bvPACeGktuSEtz6G526UBfCwpia1354fp5bYJwP2rhhnxqYkRcvTMFJ2ouddX"
iex(3)> Base58.encode(0x1777c7ba65e23151ec09125011dd25c28998c70230e7b89ca6)
"AShDKgLSuCjGZr8Fs5SRLSYvmcSV7S4zwX"

如果您已经在字符串中包含此数据,则可以使用Integer.parse/2将其转换为整数:

iex(5)> with {integer, _} <- Integer.parse("1777c7ba65e23151ec09125011dd25c28998c70230e7b89ca6", 16) do
...(5)>   Base58.encode(integer)
...(5)> end
"AShDKgLSuCjGZr8Fs5SRLSYvmcSV7S4zwX"