forked from boostorg/beast
file_win32 supports UTF-8 paths:
fix #793, close #1791, close #1793 This brings file_win32 in sync with the documentation. Previously, the path passed to open worked if encoded in the system codepage (which is almost never UTF-8). Now, the path must be encoded as UTF-8, as stated in the documentation. Adapt file tests so that for file_win32 all paths include a unicorn character.
This commit is contained in:
committed by
Vinnie Falco
parent
7912e2bef2
commit
f0c92f472c
@@ -2,6 +2,7 @@ Version 282:
|
|||||||
|
|
||||||
* Use superproject docca
|
* Use superproject docca
|
||||||
* Fix release build of docs
|
* Fix release build of docs
|
||||||
|
* file_win32 supports UTF-8 paths
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
82
include/boost/beast/core/detail/win32_unicode_path.hpp
Normal file
82
include/boost/beast/core/detail/win32_unicode_path.hpp
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
//
|
||||||
|
// Copyright (c) 2019 Mika Fischer (mika.fischer@zoopnet.de)
|
||||||
|
//
|
||||||
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
||||||
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
||||||
|
//
|
||||||
|
// Official repository: https://github.com/boostorg/beast
|
||||||
|
//
|
||||||
|
|
||||||
|
#ifndef BOOST_BEAST_CORE_DETAIL_WIN32_UNICODE_PATH_HPP
|
||||||
|
#define BOOST_BEAST_CORE_DETAIL_WIN32_UNICODE_PATH_HPP
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
#include <boost/config.hpp>
|
||||||
|
#include <boost/beast/core/error.hpp>
|
||||||
|
#include <boost/winapi/character_code_conversion.hpp>
|
||||||
|
#include <boost/winapi/file_management.hpp>
|
||||||
|
#include <boost/winapi/get_last_error.hpp>
|
||||||
|
#include <array>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace boost {
|
||||||
|
namespace beast {
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
class win32_unicode_path
|
||||||
|
{
|
||||||
|
using WCHAR_ = boost::winapi::WCHAR_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
win32_unicode_path(const char* utf8_path, error_code& ec) {
|
||||||
|
int ret = mb2wide(utf8_path, static_buf_.data(),
|
||||||
|
static_buf_.size());
|
||||||
|
if (ret == 0)
|
||||||
|
{
|
||||||
|
int sz = mb2wide(utf8_path, nullptr, 0);
|
||||||
|
if (sz == 0)
|
||||||
|
{
|
||||||
|
ec.assign(boost::winapi::GetLastError(),
|
||||||
|
system_category());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
dynamic_buf_.resize(sz);
|
||||||
|
int ret2 = mb2wide(utf8_path,
|
||||||
|
dynamic_buf_.data(),
|
||||||
|
dynamic_buf_.size());
|
||||||
|
if (ret2 == 0)
|
||||||
|
{
|
||||||
|
ec.assign(boost::winapi::GetLastError(),
|
||||||
|
system_category());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WCHAR_ const* c_str() const noexcept
|
||||||
|
{
|
||||||
|
return dynamic_buf_.empty()
|
||||||
|
? static_buf_.data()
|
||||||
|
: dynamic_buf_.data();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
int mb2wide(const char* utf8_path, WCHAR_* buf, size_t sz)
|
||||||
|
{
|
||||||
|
return boost::winapi::MultiByteToWideChar(
|
||||||
|
boost::winapi::CP_UTF8_,
|
||||||
|
boost::winapi::MB_ERR_INVALID_CHARS_,
|
||||||
|
utf8_path, -1,
|
||||||
|
buf, static_cast<int>(sz));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<WCHAR_, boost::winapi::MAX_PATH_> static_buf_;
|
||||||
|
std::vector<WCHAR_> dynamic_buf_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // detail
|
||||||
|
} // beast
|
||||||
|
} // boost
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
@@ -14,10 +14,10 @@
|
|||||||
|
|
||||||
#if BOOST_BEAST_USE_WIN32_FILE
|
#if BOOST_BEAST_USE_WIN32_FILE
|
||||||
|
|
||||||
|
#include <boost/beast/core/detail/win32_unicode_path.hpp>
|
||||||
#include <boost/core/exchange.hpp>
|
#include <boost/core/exchange.hpp>
|
||||||
#include <boost/winapi/access_rights.hpp>
|
#include <boost/winapi/access_rights.hpp>
|
||||||
#include <boost/winapi/error_codes.hpp>
|
#include <boost/winapi/error_codes.hpp>
|
||||||
#include <boost/winapi/file_management.hpp>
|
|
||||||
#include <boost/winapi/get_last_error.hpp>
|
#include <boost/winapi/get_last_error.hpp>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
@@ -186,8 +186,12 @@ open(char const* path, file_mode mode, error_code& ec)
|
|||||||
flags_and_attributes = 0x08000000; // FILE_FLAG_SEQUENTIAL_SCAN
|
flags_and_attributes = 0x08000000; // FILE_FLAG_SEQUENTIAL_SCAN
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
h_ = ::CreateFileA(
|
|
||||||
path,
|
detail::win32_unicode_path unicode_path(path, ec);
|
||||||
|
if (ec)
|
||||||
|
return;
|
||||||
|
h_ = ::CreateFileW(
|
||||||
|
unicode_path.c_str(),
|
||||||
desired_access,
|
desired_access,
|
||||||
share_mode,
|
share_mode,
|
||||||
NULL,
|
NULL,
|
||||||
|
@@ -15,6 +15,7 @@
|
|||||||
#include <boost/beast/core/file_base.hpp>
|
#include <boost/beast/core/file_base.hpp>
|
||||||
#include <boost/filesystem/path.hpp>
|
#include <boost/filesystem/path.hpp>
|
||||||
#include <boost/filesystem/operations.hpp>
|
#include <boost/filesystem/operations.hpp>
|
||||||
|
#include <boost/filesystem/fstream.hpp>
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
@@ -22,7 +23,7 @@
|
|||||||
namespace boost {
|
namespace boost {
|
||||||
namespace beast {
|
namespace beast {
|
||||||
|
|
||||||
template<class File>
|
template<class File, bool append_unicode_suffix = false>
|
||||||
void
|
void
|
||||||
test_file()
|
test_file()
|
||||||
{
|
{
|
||||||
@@ -35,16 +36,38 @@ test_file()
|
|||||||
|
|
||||||
namespace fs = boost::filesystem;
|
namespace fs = boost::filesystem;
|
||||||
|
|
||||||
|
static constexpr
|
||||||
|
#ifdef _WIN32
|
||||||
|
boost::winapi::WCHAR_ unicode_suffix[] = { 0xd83e, 0xdd84, 0x0000 }; // UTF-16-LE unicorn
|
||||||
|
#else
|
||||||
|
char unicode_suffix[] = { 0xf0, 0x9f, 0xa6, 0x84, 0x00 }; // UTF-8 unicorn
|
||||||
|
#endif
|
||||||
|
|
||||||
class temp_path
|
class temp_path
|
||||||
{
|
{
|
||||||
fs::path path_;
|
fs::path path_;
|
||||||
std::string str_;
|
std::vector<char> utf8_str_;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
temp_path()
|
temp_path()
|
||||||
: path_(fs::unique_path())
|
: path_(fs::unique_path())
|
||||||
, str_(path_.string<std::string>())
|
|
||||||
{
|
{
|
||||||
|
if (append_unicode_suffix)
|
||||||
|
path_ += unicode_suffix;
|
||||||
|
#ifdef _WIN32
|
||||||
|
constexpr auto cp = boost::winapi::CP_UTF8_;
|
||||||
|
constexpr auto flags = boost::winapi::WC_ERR_INVALID_CHARS_;
|
||||||
|
auto sz = boost::winapi::WideCharToMultiByte(
|
||||||
|
cp, flags, path_.c_str(), -1, nullptr, 0,
|
||||||
|
nullptr, nullptr);
|
||||||
|
BEAST_EXPECT(sz != 0);
|
||||||
|
utf8_str_.resize(sz);
|
||||||
|
auto ret = boost::winapi::WideCharToMultiByte(
|
||||||
|
cp, flags, path_.c_str(), -1,
|
||||||
|
utf8_str_.data(), sz,
|
||||||
|
nullptr, nullptr);
|
||||||
|
BEAST_EXPECT(ret == sz);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
operator fs::path const&()
|
operator fs::path const&()
|
||||||
@@ -54,25 +77,27 @@ test_file()
|
|||||||
|
|
||||||
operator char const*()
|
operator char const*()
|
||||||
{
|
{
|
||||||
return str_.c_str();
|
#ifdef _WIN32
|
||||||
|
return utf8_str_.data();
|
||||||
|
#else
|
||||||
|
return path_.c_str();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
auto const create =
|
auto const create =
|
||||||
[](fs::path const& path)
|
[](fs::path const& path)
|
||||||
{
|
{
|
||||||
auto const s =
|
|
||||||
path.string<std::string>();
|
|
||||||
BEAST_EXPECT(! fs::exists(path));
|
BEAST_EXPECT(! fs::exists(path));
|
||||||
FILE* f = ::fopen(s.c_str(), "w");
|
fs::ofstream out(path);
|
||||||
if( BEAST_EXPECT(f != nullptr))
|
BEAST_EXPECT(out.is_open());
|
||||||
::fclose(f);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
auto const remove =
|
auto const remove =
|
||||||
[](fs::path const& path)
|
[](fs::path const& path)
|
||||||
{
|
{
|
||||||
fs::remove(path);
|
fs::remove(path);
|
||||||
|
BEAST_EXPECT(! fs::exists(path));
|
||||||
};
|
};
|
||||||
|
|
||||||
temp_path path;
|
temp_path path;
|
||||||
|
@@ -26,7 +26,7 @@ public:
|
|||||||
void
|
void
|
||||||
run()
|
run()
|
||||||
{
|
{
|
||||||
test_file<file_win32>();
|
test_file<file_win32, true>();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user