blob: 990321b0ef4613ec60a2e83d3b7acc77a7373c9c [file] [log] [blame]
/**
* Copyright 2010 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// Author: jmarantz@google.com (Joshua Marantz)
#include "net/instaweb/util/public/string_util.h"
#include <vector>
namespace net_instaweb {
void SplitStringPieceToVector(const StringPiece& sp, const char* separator,
std::vector<StringPiece>* components,
bool omit_empty_strings) {
size_t prev_pos = 0;
size_t pos = 0;
StringPiece sep(separator);
while ((pos = sp.find(sep, pos)) != StringPiece::npos) {
if (!omit_empty_strings || (pos > prev_pos)) {
components->push_back(sp.substr(prev_pos, pos - prev_pos));
}
++pos;
prev_pos = pos;
}
if (!omit_empty_strings || (prev_pos < sp.size())) {
components->push_back(sp.substr(prev_pos, prev_pos - sp.size()));
}
}
std::string StrCat(const StringPiece& a,
const StringPiece& b,
const StringPiece& c,
const StringPiece& d,
const StringPiece& e,
const StringPiece& f,
const StringPiece& g,
const StringPiece& h) {
std::string res;
res.reserve(a.size() + b.size() + c.size() + d.size() + e.size() + f.size() +
g.size() + h.size());
a.AppendToString(&res);
b.AppendToString(&res);
c.AppendToString(&res);
d.AppendToString(&res);
e.AppendToString(&res);
f.AppendToString(&res);
g.AppendToString(&res);
h.AppendToString(&res);
return res;
}
void BackslashEscape(const StringPiece& src,
const StringPiece& to_escape,
std::string* dest) {
dest->reserve(dest->size() + src.size());
for (const char *p = src.data(), *end = src.data() + src.size();
p != end; ++p) {
if (to_escape.find(*p) != StringPiece::npos) {
dest->push_back('\\');
}
dest->push_back(*p);
}
}
bool StringCaseEqual(const StringPiece& s1, const StringPiece& s2) {
return (s1.size() == s2.size() &&
0 == strncasecmp(s1.data(), s2.data(), s1.size()));
}
bool StringCaseStartsWith(const StringPiece& str, const StringPiece& prefix) {
return (str.size() >= prefix.size() &&
0 == strncasecmp(str.data(), prefix.data(), prefix.size()));
}
bool StringCaseEndsWith(const StringPiece& str, const StringPiece& suffix) {
return (str.size() >= suffix.size() &&
0 == strncasecmp(str.data() + str.size() - suffix.size(),
suffix.data(), suffix.size()));
}
void ParseShellLikeString(const StringPiece& input,
std::vector<std::string>* output) {
output->clear();
for (size_t index = 0; index < input.size();) {
const char ch = input[index];
// If we see a quoted section, treat it as a single item even if there are
// spaces in it.
if (ch == '"' || ch == '\'') {
const char quote = ch;
++index; // skip open quote
output->push_back("");
std::string& part = output->back();
for (; index < input.size() && input[index] != quote; ++index) {
if (input[index] == '\\') {
++index; // skip backslash
if (index >= input.size()) {
break;
}
}
part.push_back(input[index]);
}
++index; // skip close quote
}
// Without quotes, items are whitespace-separated.
else if (!isspace(ch)) {
output->push_back("");
std::string& part = output->back();
for (;index < input.size() && !isspace(input[index]); ++index) {
part.push_back(input[index]);
}
}
// Ignore whitespace (outside of quotes).
else {
++index;
}
}
}
// From src/third_party/protobuf2/src/src/google/protobuf/stubs/strutil.h
// but we don't need any other aspect of protobufs so we don't want to
// incur the link cost.
void LowerString(std::string* s) {
std::string::iterator end = s->end();
for (std::string::iterator i = s->begin(); i != end; ++i) {
// tolower() changes based on locale. We don't want this!
if ('A' <= *i && *i <= 'Z') *i += 'a' - 'A';
}
}
// From src/third_party/protobuf2/src/src/google/protobuf/stubs/strutil.h
// but we don't need any other aspect of protobufs so we don't want to
// incur the link cost.
bool HasPrefixString(const StringPiece& str, const StringPiece& prefix) {
return ((str.size() >= prefix.size()) &&
(str.substr(0, prefix.size()) == prefix));
}
const StringPiece EmptyString::kEmptyString;
} // namespace net_instaweb