You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

165 lines
5.2 KiB

/*
* Copyright 2002-2019 Intel Corporation.
*
* This software and the related documents are Intel copyrighted materials, and your
* use of them is governed by the express license under which they were provided to
* you ("License"). Unless the License provides otherwise, you may not use, modify,
* copy, publish, distribute, disclose or transmit this software or the related
* documents without Intel's prior written permission.
*
* This software and the related documents are provided as is, with no express or
* implied warranties, other than those that are expressly stated in the License.
*/
// <COMPONENT>: util
// <FILE-TYPE>: component public header
#ifndef UTIL_QUOTE_ARGUMENT_MS_HPP
#define UTIL_QUOTE_ARGUMENT_MS_HPP
namespace UTIL {
/*!
* This is a base class. Use QUOTE_ARGUMENT_MS or QUOTE_ARGUMENT_MS_WIDE.
*/
template <typename T> class /*<UTILITY>*/ QUOTE_ARGUMENT_MS_BASE
{
private:
typedef std::basic_string<T> STRING;
STRING _quoted;
protected:
/*!
* @param[in] arg The string that needs to be quoted.
* @param[in] whitespace The set of characters that are considered whitespace.
*/
QUOTE_ARGUMENT_MS_BASE(const STRING &arg, const T *whitespace)
{
// Quoting is only necessary if the argument contains whitespace or a quote (").
//
_quoted = arg;
if (_quoted.find_first_of(whitespace) != STRING::npos ||
_quoted.find_first_of('"') != STRING::npos)
{
EscapeBackSlashes();
EscapeQuotes();
AddQuotes(whitespace);
}
}
/*!
* @return The quoted string.
*/
STRING Get() {return _quoted;}
private:
/*!
* Add escaping to each sequence of backslashes that immediately precedes
* a quote character. Each backslash in such a sequence must be escaped
* with another backslash. Note that backslashes that are not part of a
* sequence that immediately precedes a quote are NOT escaped. E.g.:
*
* \a" -> \a"
* foo\bar"baz\\"fum -> foo\bar"baz\\\\"fum
*/
void EscapeBackSlashes()
{
size_t quote = _quoted.find_first_of('"', 1);
while (quote != STRING::npos)
{
size_t numSlashes = 0;
size_t notSlash = _quoted.find_last_not_of('\\', quote-1);
if (notSlash != quote-1)
{
if (notSlash == STRING::npos)
numSlashes = quote;
else
numSlashes = quote - notSlash - 1;
_quoted.insert(quote, numSlashes, '\\');
}
quote = _quoted.find_first_of('"', quote + numSlashes + 1);
}
}
/*!
* Add escaping to each quote character (") by preceding it with a backslash (\).
*/
void EscapeQuotes()
{
size_t quote = _quoted.find_first_of('"');
while (quote != STRING::npos)
{
_quoted.insert(quote, 1, '\\');
quote = _quoted.find_first_of('"', quote + 2);
}
}
/*!
* Add quotes around the string if it contains any whitespace.
*/
void AddQuotes(const T *whitespace)
{
if (_quoted.find_first_of(whitespace) != STRING::npos)
{
_quoted.insert((size_t)0, (size_t)1, '"');
_quoted.append(1, '"');
// If the last character (prior to adding the quotes) was a backslash,
// it needs to be escaped now because it precedes a quote.
//
size_t quote = _quoted.size() - 1;
if (_quoted[quote-1] == '\\')
{
size_t notSlash = _quoted.find_last_not_of('\\', quote-2);
size_t numSlashes = quote - notSlash - 1;
_quoted.insert(quote, numSlashes, '\\');
}
}
}
};
/*!
* Utility that adds quoting to a string that is necessary in order to pass it
* as an argument to a C/C++ program on Microsoft Windows. The quoting handles
* internal spaces, embedded quote characters, etc.
*/
class /*<UTILITY>*/ QUOTE_ARGUMENT_MS : private QUOTE_ARGUMENT_MS_BASE<char>
{
public:
/*!
* @param[in] arg The string that needs to be quoted.
* @param[in] whitespace The set of characters that are considered whitespace.
*/
QUOTE_ARGUMENT_MS(std::string str, const char *ws = " \t") : QUOTE_ARGUMENT_MS_BASE<char>(str, ws) {}
/*!
* @return The quoted string.
*/
std::string Get() {return QUOTE_ARGUMENT_MS_BASE<char>::Get();}
};
/*!
* A wide-character version of QUOTE_ARGUMENT_MS. Adds quoting to a string, allowing
* it to be passed as an argument to a C/C++ program on Microsoft Windows.
*/
class /*<UTILITY>*/ QUOTE_ARGUMENT_MS_WIDE : private QUOTE_ARGUMENT_MS_BASE<wchar_t>
{
public:
/*!
* @param[in] arg The string that needs to be quoted.
* @param[in] whitespace The set of characters that are considered whitespace.
*/
QUOTE_ARGUMENT_MS_WIDE(std::wstring str, const wchar_t *ws = L" \t") : QUOTE_ARGUMENT_MS_BASE<wchar_t>(str, ws) {}
/*!
* @return The quoted string.
*/
std::wstring Get() {return QUOTE_ARGUMENT_MS_BASE<wchar_t>::Get();}
};
} // namespace
#endif // file guard