22#include <boost/asio/streambuf.hpp>
23#include <boost/asio/read.hpp>
24#include <boost/asio/read_until.hpp>
25#include <boost/asio/write.hpp>
26#include <boost/asio/ssl/stream.hpp>
27#include <boost/filesystem/operations.hpp>
34 case ERROR_SUCCESS:
return "SUCCESS";
35 case ERROR_FILE_CORRUPT:
return "FILE_CORRUPT";
36 case ERROR_CAN_NOT_COMPLETE:
return "CAN_NOT_COMPLETE";
37 case ERROR_HANDLE_EOF:
return "HANDLE_EOF";
38 case ERROR_NO_MORE_FILES:
return "NO_MORE_FILES";
39 case ERROR_BAD_FORMAT:
return "BAD_FORMAT";
40 case ERROR_INSUFFICIENT_BUFFER:
return "INSUFFICIENT_BUFFER";
41 case ERROR_ALREADY_EXISTS:
return "ALREADY_EXISTS";
42 case ERROR_DISK_FULL:
return "DISK_FULL";
43 case ERROR_INVALID_PARAMETER:
return "INVALID_PARAMETER";
44 case ERROR_NOT_SUPPORTED:
return "NOT_SUPPORTED";
45 case ERROR_NOT_ENOUGH_MEMORY:
return "NOT_ENOUGH_MEMORY";
46 case ERROR_INVALID_HANDLE:
return "INVALID_HANDLE";
47 case ERROR_ACCESS_DENIED:
return "ACCESS_DENIED";
48 case ERROR_FILE_NOT_FOUND:
return "FILE_NOT_FOUND";
49 case ERROR_FILE_ENCRYPTED:
return "FILE_ENCRYPTED";
50 case ERROR_FILE_OFFLINE:
return "FILE_OFFLINE";
51 default:
return "UNKNOWN";
59 boost::system::error_code error;
61 boost::asio::ssl::context sslContext(boost::asio::ssl::context::sslv23);
62 sslContext.set_options(boost::asio::ssl::context::no_sslv2, error);
63 sslContext.set_options(boost::asio::ssl::context::no_sslv3, error);
64 sslContext.set_options(boost::asio::ssl::context::no_tlsv1, error);
65 sslContext.set_options(boost::asio::ssl::context::no_tlsv1_1, error);
66 sslContext.set_default_verify_paths(error);
74 boost::asio::ssl::stream<boost::asio::ip::tcp::socket> socket(ioContext, sslContext);
75 socket.set_verify_mode(boost::asio::ssl::verify_none, error);
79 socket.lowest_layer().connect(*endpoint, error);
83 if (!SSL_set_tlsext_host_name(socket.native_handle(), serverName.c_str()))
86 socket.handshake(boost::asio::ssl::stream_base::client, error);
90 boost::asio::streambuf request;
91 std::ostream request_stream(&request);
93 request_stream <<
"GET " << getCommand <<
" HTTP/1.0\r\n";
94 request_stream <<
"Host: " << serverName <<
"\r\n";
95 request_stream <<
"Connection: close\r\n\r\n";
98 boost::asio::write(socket, request);
101 boost::asio::streambuf response;
102 boost::asio::read_until(socket, response,
"\r\n", error);
105 printf(
"Downloading tact key list failed to read HTTP response status %s", error.message().c_str());
110 std::string http_version;
112 std::string status_message;
113 std::istream response_stream(&response);
115 response_stream >> http_version;
116 response_stream >> status_code;
117 std::getline(response_stream, status_message);
119 if (status_code != 200)
121 printf(
"Downloading tact key list failed with server response %u %s", status_code, status_message.c_str());
126 boost::asio::read_until(socket, response,
"\r\n\r\n");
129 printf(
"Downloading tact key list failed to read HTTP response headers %s", error.message().c_str());
135 while (std::getline(response_stream, header) && header !=
"\r")
139 std::stringstream rawBody;
142 if (response.size() > 0)
143 rawBody << &response;
146 while (boost::asio::read(socket, response, boost::asio::transfer_at_least(1), error))
147 rawBody << &response;
149 return rawBody.str();
153 bool GetStorageInfo(HANDLE storage, CASC_STORAGE_INFO_CLASS storageInfoClass, T* value)
155 size_t infoDataSizeNeeded = 0;
156 return ::CascGetStorageInfo(storage, storageInfoClass, value,
sizeof(T), &infoDataSizeNeeded);
167 static Optional<std::string> const tactKeys = DownloadFile(
"raw.githubusercontent.com", 443,
"/wowdev/TACTKeys/master/WoW.txt");
169 return tactKeys && CascImportKeysFromString(_handle, tactKeys->c_str());
174 ::CascCloseStorage(_handle);
179 std::string strPath = path.string();
180 CASC_OPEN_STORAGE_ARGS args = {};
181 args.Size =
sizeof(CASC_OPEN_STORAGE_ARGS);
182 args.szLocalPath = strPath.c_str();
183 args.szCodeName = product;
184 args.dwLocaleMask = localeMask;
185 HANDLE handle =
nullptr;
186 if (!::CascOpenStorageEx(
nullptr, &args,
false, &handle))
188 DWORD lastError = GetCascError();
189 printf(
"Error opening casc storage '%s': %s\n", path.string().c_str(),
HumanReadableCASCError(lastError));
190 CascCloseStorage(handle);
191 SetCascError(lastError);
195 printf(
"Opened casc storage '%s'\n", path.string().c_str());
199 printf(
"Failed to load additional online encryption keys, some files might not be extracted.\n");
206 std::string strPath = path.string();
207 CASC_OPEN_STORAGE_ARGS args = {};
208 args.Size =
sizeof(CASC_OPEN_STORAGE_ARGS);
209 args.szLocalPath = strPath.c_str();
210 args.szCodeName = product;
211 args.szRegion = region;
212 args.dwLocaleMask = localeMask;
214 HANDLE handle =
nullptr;
215 if (!::CascOpenStorageEx(
nullptr, &args,
true, &handle))
217 DWORD lastError = GetCascError();
219 CascCloseStorage(handle);
220 SetCascError(lastError);
225 if (!GetStorageInfo(handle, CascStorageFeatures, &features) || !(features & CASC_FEATURE_ONLINE))
227 printf(
"Local casc storage detected in cache path \"%s\" (or its parent directory). Remote storage not opened!\n", args.szLocalPath);
228 CascCloseStorage(handle);
229 SetCascError(ERROR_FILE_OFFLINE);
233 printf(
"Opened remote casc storage '%s'\n", path.string().c_str());
237 printf(
"Failed to load additional online encryption keys, some files might not be extracted.\n");
244 CASC_STORAGE_PRODUCT product;
245 if (GetStorageInfo(_handle, CascStorageProduct, &product))
246 return product.BuildNumber;
254 if (GetStorageInfo(_handle, CascStorageInstalledLocales, &locales))
262 return CascFindEncryptionKey(_handle, keyLookup) !=
nullptr;
267 DWORD openFlags = CASC_OPEN_BY_NAME;
268 if (zerofillEncryptedParts)
269 openFlags |= CASC_OVERCOME_ENCRYPTED;
271 HANDLE handle =
nullptr;
272 if (!::CascOpenFile(_handle, fileName, localeMask, openFlags, &handle))
274 DWORD lastError = GetCascError();
276 fprintf(stderr,
"Failed to open '%s' in CASC storage: %s\n", fileName,
HumanReadableCASCError(lastError));
278 CascCloseFile(handle);
279 SetCascError(lastError);
283 return new File(handle);
288 DWORD openFlags = CASC_OPEN_BY_FILEID;
289 if (zerofillEncryptedParts)
290 openFlags |= CASC_OVERCOME_ENCRYPTED;
292 HANDLE handle =
nullptr;
293 if (!::CascOpenFile(_handle, CASC_FILE_DATA_ID(fileDataId), localeMask, openFlags, &handle))
295 DWORD lastError = GetCascError();
297 fprintf(stderr,
"Failed to open 'FileDataId %u' in CASC storage: %s\n", fileDataId,
HumanReadableCASCError(lastError));
299 CascCloseFile(handle);
300 SetCascError(lastError);
304 return new File(handle);
313 ::CascCloseFile(_handle);
318 CASC_FILE_FULL_INFO info;
319 if (!::CascGetFileInfo(_handle, CascFileFullInfo, &info,
sizeof(info),
nullptr))
320 return CASC_INVALID_ID;
322 return info.FileDataId;
328 if (!::CascGetFileSize64(_handle, &
size))
337 if (!::CascSetFilePointer64(_handle, 0, &position, FILE_CURRENT))
340 return int64(position);
346 memcpy(parts, &position,
sizeof(parts));
347 return ::CascSetFilePointer64(_handle, position,
nullptr, FILE_BEGIN);
352 DWORD bytesReadDWORD;
353 if (!::CascReadFile(_handle, buffer, bytes, &bytesReadDWORD))
357 *bytesRead = bytesReadDWORD;
std::optional< T > Optional
Optional helper class to wrap optional values within.
bool SetPointer(int64 position)
bool ReadFile(void *buffer, uint32 bytes, uint32 *bytesRead)
bool HasTactKey(uint64 keyLookup) const
uint32 GetBuildNumber() const
static Storage * OpenRemote(boost::filesystem::path const &path, uint32 localeMask, char const *product, char const *region)
static Storage * Open(boost::filesystem::path const &path, uint32 localeMask, char const *product)
uint32 GetInstalledLocalesMask() const
File * OpenFile(char const *fileName, uint32 localeMask, bool printErrors=false, bool zerofillEncryptedParts=false) const
bool LoadOnlineTactKeys()
char const * HumanReadableCASCError(uint32 error)
constexpr std::size_t size()