forked from trustwallet/wallet-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHexCoding.h
142 lines (127 loc) · 3.49 KB
/
HexCoding.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
// SPDX-License-Identifier: Apache-2.0
//
// Copyright © 2017 Trust Wallet.
#pragma once
#include "Data.h"
#include "rust/bindgen/WalletCoreRSBindgen.h"
#include "rust/Wrapper.h"
#include <algorithm>
#include <array>
#include <sstream>
#include <string>
#include <tuple>
namespace TW::internal {
/// Parses a string of hexadecimal values.
///
/// \returns the array or parsed bytes or an empty array if the string is not
/// valid hexadecimal.
inline Data parse_hex(const std::string& input) {
if (input.empty()) {
return {};
}
Rust::CByteArrayResultWrapper res = Rust::decode_hex(input.c_str());
return res.unwrap_or_default().data;
}
}
namespace TW {
inline bool is_hex_encoded(const std::string& s)
{
bool with_0x = s.compare(0, 2, "0x") == 0
&& s.size() > 2
&& s.find_first_not_of("0123456789abcdefABCDEF", 2) == std::string::npos;
bool without_0x = s.find_first_not_of("0123456789abcdefABCDEF") == std::string::npos;
return with_0x || without_0x;
}
/// Converts a collection of bytes to a hexadecimal string representation.
template <typename T>
inline std::string hex(const T& collection, bool prefixed = false) {
auto rust_functor = [prefixed](auto&& collection){
auto res = Rust::encode_hex(collection.data(), collection.size(), prefixed);
std::string encoded_str(res);
Rust::free_string(res);
return encoded_str;
};
if constexpr (std::is_same_v<T, Data>) {
return rust_functor(collection);
}
else if constexpr (std::is_same_v<T, std::string>) {
return rust_functor(data(collection));
}
else {
return rust_functor(data_from(collection));
}
}
/// same as hex, with 0x prefix
template <typename T>
inline std::string hexEncoded(T&& collection) {
return hex(std::forward<T>(collection), true);
}
/// Converts a `uint64_t` value to a hexadecimal string.
inline std::string hex(uint64_t value) {
const uint8_t* begin = reinterpret_cast<const uint8_t*>(&value);
const uint8_t* end = begin + sizeof(value);
Data v(begin, end);
std::reverse(v.begin(), v.end());
return hex(v);
}
/// Parses a string of hexadecimal values.
///
/// \returns the array or parsed bytes or an empty array if the string is not
/// valid hexadecimal.
inline Data parse_hex(const std::string& string, bool padLeft = false) {
if (string.size() % 2 != 0 && padLeft) {
std::string temp = string;
if (temp.compare(0, 2, "0x") == 0) {
temp.erase(0, 2);
}
temp.insert(0, 1, '0');
return internal::parse_hex(temp);
}
return internal::parse_hex(string);
}
inline const char* hex_char_to_bin(char c) {
switch (toupper(c)) {
case '0':
return "0000";
case '1':
return "0001";
case '2':
return "0010";
case '3':
return "0011";
case '4':
return "0100";
case '5':
return "0101";
case '6':
return "0110";
case '7':
return "0111";
case '8':
return "1000";
case '9':
return "1001";
case 'A':
return "1010";
case 'B':
return "1011";
case 'C':
return "1100";
case 'D':
return "1101";
case 'E':
return "1110";
case 'F':
return "1111";
default:
return "";
}
}
inline std::string hex_str_to_bin_str(const std::string& hex) {
std::stringstream ss;
for (auto&& c: hex) {
ss << hex_char_to_bin(c);
}
return ss.str();
}
} // namespace TW