|
|
@ -28,10 +28,18 @@
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
|
|
|
#include <io.h>
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#endif
|
|
|
|
#include <climits>
|
|
|
|
#include <climits>
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <fstream>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
|
|
#include <google/protobuf/stubs/hash.h>
|
|
|
|
#include <google/protobuf/stubs/hash.h>
|
|
|
@ -39,6 +47,7 @@
|
|
|
|
#include <google/protobuf/io/coded_stream.h>
|
|
|
|
#include <google/protobuf/io/coded_stream.h>
|
|
|
|
#include <google/protobuf/io/zero_copy_stream_impl.h>
|
|
|
|
#include <google/protobuf/io/zero_copy_stream_impl.h>
|
|
|
|
#include <google/protobuf/descriptor.pb.h>
|
|
|
|
#include <google/protobuf/descriptor.pb.h>
|
|
|
|
|
|
|
|
#include <google/protobuf/stubs/common.h>
|
|
|
|
#include <google/protobuf/stubs/strutil.h>
|
|
|
|
#include <google/protobuf/stubs/strutil.h>
|
|
|
|
|
|
|
|
|
|
|
|
// NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
|
|
|
|
// NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
|
|
|
@ -51,45 +60,6 @@ namespace objectivec {
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
|
|
// islower()/isupper()/tolower()/toupper() change based on locale.
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
// src/google/protobuf/stubs/strutil.h:150 has the same pattern. For the
|
|
|
|
|
|
|
|
// Objective C plugin, test failures were seen on TravisCI because isupper('A')
|
|
|
|
|
|
|
|
// was coming back false for some server's locale. This approach avoids any
|
|
|
|
|
|
|
|
// such issues.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool IsLower(const char c) {
|
|
|
|
|
|
|
|
return ('a' <= c && c <= 'z');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool IsUpper(const char c) {
|
|
|
|
|
|
|
|
return ('A' <= c && c <= 'Z');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char ToLower(char c) {
|
|
|
|
|
|
|
|
if ('A' <= c && c <= 'Z') {
|
|
|
|
|
|
|
|
c += 'a' - 'A';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// toupper() changes based on locale. We don't want this!
|
|
|
|
|
|
|
|
char ToUpper(char c) {
|
|
|
|
|
|
|
|
if ('a' <= c && c <= 'z') {
|
|
|
|
|
|
|
|
c += 'A' - 'a';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string TrimString(const string& s) {
|
|
|
|
|
|
|
|
string::size_type start = s.find_first_not_of(" \n\r\t");
|
|
|
|
|
|
|
|
if (start == string::npos) {
|
|
|
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
string::size_type end = s.find_last_not_of(" \n\r\t") + 1;
|
|
|
|
|
|
|
|
return s.substr(start, end - start);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
hash_set<string> MakeWordsMap(const char* const words[], size_t num_words) {
|
|
|
|
hash_set<string> MakeWordsMap(const char* const words[], size_t num_words) {
|
|
|
|
hash_set<string> result;
|
|
|
|
hash_set<string> result;
|
|
|
|
for (int i = 0; i < num_words; i++) {
|
|
|
|
for (int i = 0; i < num_words; i++) {
|
|
|
@ -115,7 +85,7 @@ string UnderscoresToCamelCase(const string& input, bool first_capitalized) {
|
|
|
|
bool last_char_was_upper = false;
|
|
|
|
bool last_char_was_upper = false;
|
|
|
|
for (int i = 0; i < input.size(); i++) {
|
|
|
|
for (int i = 0; i < input.size(); i++) {
|
|
|
|
char c = input[i];
|
|
|
|
char c = input[i];
|
|
|
|
if (c >= '0' && c <= '9') {
|
|
|
|
if (ascii_isdigit(c)) {
|
|
|
|
if (!last_char_was_number) {
|
|
|
|
if (!last_char_was_number) {
|
|
|
|
values.push_back(current);
|
|
|
|
values.push_back(current);
|
|
|
|
current = "";
|
|
|
|
current = "";
|
|
|
@ -123,7 +93,7 @@ string UnderscoresToCamelCase(const string& input, bool first_capitalized) {
|
|
|
|
current += c;
|
|
|
|
current += c;
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_number = true;
|
|
|
|
last_char_was_number = true;
|
|
|
|
} else if (IsLower(c)) {
|
|
|
|
} else if (ascii_islower(c)) {
|
|
|
|
// lowercase letter can follow a lowercase or uppercase letter
|
|
|
|
// lowercase letter can follow a lowercase or uppercase letter
|
|
|
|
if (!last_char_was_lower && !last_char_was_upper) {
|
|
|
|
if (!last_char_was_lower && !last_char_was_upper) {
|
|
|
|
values.push_back(current);
|
|
|
|
values.push_back(current);
|
|
|
@ -132,12 +102,12 @@ string UnderscoresToCamelCase(const string& input, bool first_capitalized) {
|
|
|
|
current += c; // already lower
|
|
|
|
current += c; // already lower
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_lower = true;
|
|
|
|
last_char_was_lower = true;
|
|
|
|
} else if (IsUpper(c)) {
|
|
|
|
} else if (ascii_isupper(c)) {
|
|
|
|
if (!last_char_was_upper) {
|
|
|
|
if (!last_char_was_upper) {
|
|
|
|
values.push_back(current);
|
|
|
|
values.push_back(current);
|
|
|
|
current = "";
|
|
|
|
current = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
current += ToLower(c);
|
|
|
|
current += ascii_tolower(c);
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_number = last_char_was_lower = last_char_was_upper = false;
|
|
|
|
last_char_was_upper = true;
|
|
|
|
last_char_was_upper = true;
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -151,7 +121,7 @@ string UnderscoresToCamelCase(const string& input, bool first_capitalized) {
|
|
|
|
bool all_upper = (kUpperSegments.count(value) > 0);
|
|
|
|
bool all_upper = (kUpperSegments.count(value) > 0);
|
|
|
|
for (int j = 0; j < value.length(); j++) {
|
|
|
|
for (int j = 0; j < value.length(); j++) {
|
|
|
|
if (j == 0 || all_upper) {
|
|
|
|
if (j == 0 || all_upper) {
|
|
|
|
value[j] = ToUpper(value[j]);
|
|
|
|
value[j] = ascii_toupper(value[j]);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
// Nothing, already in lower.
|
|
|
|
// Nothing, already in lower.
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -163,7 +133,7 @@ string UnderscoresToCamelCase(const string& input, bool first_capitalized) {
|
|
|
|
result += *i;
|
|
|
|
result += *i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((result.length() != 0) && !first_capitalized) {
|
|
|
|
if ((result.length() != 0) && !first_capitalized) {
|
|
|
|
result[0] = ToLower(result[0]);
|
|
|
|
result[0] = ascii_tolower(result[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -272,7 +242,7 @@ bool IsSpecialName(const string& name, const string* special_names,
|
|
|
|
// If name is longer than the retained_name[i] that it matches
|
|
|
|
// If name is longer than the retained_name[i] that it matches
|
|
|
|
// the next character must be not lower case (newton vs newTon vs
|
|
|
|
// the next character must be not lower case (newton vs newTon vs
|
|
|
|
// new_ton).
|
|
|
|
// new_ton).
|
|
|
|
return !IsLower(name[length]);
|
|
|
|
return !ascii_islower(name[length]);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -342,30 +312,6 @@ string FileClassPrefix(const FileDescriptor* file) {
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ValidateObjCClassPrefix(const FileDescriptor* file) {
|
|
|
|
|
|
|
|
string prefix = file->options().objc_class_prefix();
|
|
|
|
|
|
|
|
if (prefix.length() > 0) {
|
|
|
|
|
|
|
|
// NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
|
|
|
|
|
|
|
|
// error cases, so it seems to be ok to use as a back door for errors.
|
|
|
|
|
|
|
|
if (!IsUpper(prefix[0])) {
|
|
|
|
|
|
|
|
cerr << endl
|
|
|
|
|
|
|
|
<< "protoc:0: warning: Invalid 'option objc_class_prefix = \""
|
|
|
|
|
|
|
|
<< prefix << "\";' in '" << file->name() << "';"
|
|
|
|
|
|
|
|
<< " it should start with a capital letter."
|
|
|
|
|
|
|
|
<< endl;
|
|
|
|
|
|
|
|
cerr.flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prefix.length() < 3) {
|
|
|
|
|
|
|
|
cerr << endl
|
|
|
|
|
|
|
|
<< "protoc:0: warning: Invalid 'option objc_class_prefix = \""
|
|
|
|
|
|
|
|
<< prefix << "\";' in '" << file->name() << "';"
|
|
|
|
|
|
|
|
<< " Apple recommends they should be at least 3 characters long."
|
|
|
|
|
|
|
|
<< endl;
|
|
|
|
|
|
|
|
cerr.flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string FileClassName(const FileDescriptor* file) {
|
|
|
|
string FileClassName(const FileDescriptor* file) {
|
|
|
|
string name = FileClassPrefix(file);
|
|
|
|
string name = FileClassPrefix(file);
|
|
|
|
name += UnderscoresToCamelCase(StripProto(BaseFileName(file)), true);
|
|
|
|
name += UnderscoresToCamelCase(StripProto(BaseFileName(file)), true);
|
|
|
@ -453,10 +399,10 @@ string UnCamelCaseEnumShortName(const string& name) {
|
|
|
|
string result;
|
|
|
|
string result;
|
|
|
|
for (int i = 0; i < name.size(); i++) {
|
|
|
|
for (int i = 0; i < name.size(); i++) {
|
|
|
|
char c = name[i];
|
|
|
|
char c = name[i];
|
|
|
|
if (i > 0 && c >= 'A' && c <= 'Z') {
|
|
|
|
if (i > 0 && ascii_isupper(c)) {
|
|
|
|
result += '_';
|
|
|
|
result += '_';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result += ToUpper(c);
|
|
|
|
result += ascii_toupper(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -487,7 +433,7 @@ string FieldNameCapitalized(const FieldDescriptor* field) {
|
|
|
|
// name.
|
|
|
|
// name.
|
|
|
|
string result = FieldName(field);
|
|
|
|
string result = FieldName(field);
|
|
|
|
if (result.length() > 0) {
|
|
|
|
if (result.length() > 0) {
|
|
|
|
result[0] = ToUpper(result[0]);
|
|
|
|
result[0] = ascii_toupper(result[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -511,7 +457,7 @@ string OneofNameCapitalized(const OneofDescriptor* descriptor) {
|
|
|
|
// Use the common handling and then up-case the first letter.
|
|
|
|
// Use the common handling and then up-case the first letter.
|
|
|
|
string result = OneofName(descriptor);
|
|
|
|
string result = OneofName(descriptor);
|
|
|
|
if (result.length() > 0) {
|
|
|
|
if (result.length() > 0) {
|
|
|
|
result[0] = ToUpper(result[0]);
|
|
|
|
result[0] = ascii_toupper(result[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -526,8 +472,8 @@ string UnCamelCaseFieldName(const string& name, const FieldDescriptor* field) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (field->type() == FieldDescriptor::TYPE_GROUP) {
|
|
|
|
if (field->type() == FieldDescriptor::TYPE_GROUP) {
|
|
|
|
if (worker.length() > 0) {
|
|
|
|
if (worker.length() > 0) {
|
|
|
|
if (worker[0] >= 'a' && worker[0] <= 'z') {
|
|
|
|
if (ascii_islower(worker[0])) {
|
|
|
|
worker[0] = ToUpper(worker[0]);
|
|
|
|
worker[0] = ascii_toupper(worker[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return worker;
|
|
|
|
return worker;
|
|
|
@ -535,11 +481,11 @@ string UnCamelCaseFieldName(const string& name, const FieldDescriptor* field) {
|
|
|
|
string result;
|
|
|
|
string result;
|
|
|
|
for (int i = 0; i < worker.size(); i++) {
|
|
|
|
for (int i = 0; i < worker.size(); i++) {
|
|
|
|
char c = worker[i];
|
|
|
|
char c = worker[i];
|
|
|
|
if (c >= 'A' && c <= 'Z') {
|
|
|
|
if (ascii_isupper(c)) {
|
|
|
|
if (i > 0) {
|
|
|
|
if (i > 0) {
|
|
|
|
result += '_';
|
|
|
|
result += '_';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result += ToLower(c);
|
|
|
|
result += ascii_tolower(c);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
result += c;
|
|
|
|
result += c;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -831,6 +777,252 @@ string BuildCommentsString(const SourceLocation& location) {
|
|
|
|
return final_comments;
|
|
|
|
return final_comments;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Internal helper class that parses the expected package to prefix mappings
|
|
|
|
|
|
|
|
// file.
|
|
|
|
|
|
|
|
class Parser {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Parser(map<string, string>* inout_package_to_prefix_map)
|
|
|
|
|
|
|
|
: prefix_map_(inout_package_to_prefix_map), line_(0) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Parses a check of input, returning success/failure.
|
|
|
|
|
|
|
|
bool ParseChunk(StringPiece chunk);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Should be called to finish parsing (after all input has been provided via
|
|
|
|
|
|
|
|
// ParseChunk()). Returns success/failure.
|
|
|
|
|
|
|
|
bool Finish();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int last_line() const { return line_; }
|
|
|
|
|
|
|
|
string error_str() const { return error_str_; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
bool ParseLoop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
map<string, string>* prefix_map_;
|
|
|
|
|
|
|
|
int line_;
|
|
|
|
|
|
|
|
string error_str_;
|
|
|
|
|
|
|
|
StringPiece p_;
|
|
|
|
|
|
|
|
string leftover_;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool Parser::ParseChunk(StringPiece chunk) {
|
|
|
|
|
|
|
|
if (!leftover_.empty()) {
|
|
|
|
|
|
|
|
chunk.AppendToString(&leftover_);
|
|
|
|
|
|
|
|
p_ = StringPiece(leftover_);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
p_ = chunk;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool result = ParseLoop();
|
|
|
|
|
|
|
|
if (p_.empty()) {
|
|
|
|
|
|
|
|
leftover_.clear();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
leftover_ = p_.ToString();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool Parser::Finish() {
|
|
|
|
|
|
|
|
if (leftover_.empty()) {
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Force a newline onto the end to finish parsing.
|
|
|
|
|
|
|
|
p_ = StringPiece(leftover_ + "\n");
|
|
|
|
|
|
|
|
if (!ParseLoop()) {
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return p_.empty(); // Everything used?
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static bool ascii_isnewline(char c) { return c == '\n' || c == '\r'; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool ReadLine(StringPiece* input, StringPiece* line) {
|
|
|
|
|
|
|
|
for (int len = 0; len < input->size(); ++len) {
|
|
|
|
|
|
|
|
if (ascii_isnewline((*input)[len])) {
|
|
|
|
|
|
|
|
*line = StringPiece(input->data(), len);
|
|
|
|
|
|
|
|
++len; // advance over the newline
|
|
|
|
|
|
|
|
*input = StringPiece(input->data() + len, input->size() - len);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return false; // Ran out of input with no newline.
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void TrimWhitespace(StringPiece* input) {
|
|
|
|
|
|
|
|
while (!input->empty() && ascii_isspace(*input->data())) {
|
|
|
|
|
|
|
|
input->remove_prefix(1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!input->empty() && ascii_isspace((*input)[input->length() - 1])) {
|
|
|
|
|
|
|
|
input->remove_suffix(1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void RemoveComment(StringPiece* input) {
|
|
|
|
|
|
|
|
int offset = input->find('#');
|
|
|
|
|
|
|
|
if (offset != StringPiece::npos) {
|
|
|
|
|
|
|
|
input->remove_suffix(input->length() - offset);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool Parser::ParseLoop() {
|
|
|
|
|
|
|
|
StringPiece line;
|
|
|
|
|
|
|
|
while (ReadLine(&p_, &line)) {
|
|
|
|
|
|
|
|
++line_;
|
|
|
|
|
|
|
|
RemoveComment(&line);
|
|
|
|
|
|
|
|
TrimWhitespace(&line);
|
|
|
|
|
|
|
|
if (line.size() == 0) {
|
|
|
|
|
|
|
|
continue; // Blank line.
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int offset = line.find('=');
|
|
|
|
|
|
|
|
if (offset == StringPiece::npos) {
|
|
|
|
|
|
|
|
error_str_ =
|
|
|
|
|
|
|
|
string("Line without equal sign: '") + line.ToString() + "'.";
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
StringPiece package(line, 0, offset);
|
|
|
|
|
|
|
|
StringPiece prefix(line, offset + 1, line.length() - offset - 1);
|
|
|
|
|
|
|
|
TrimWhitespace(&package);
|
|
|
|
|
|
|
|
TrimWhitespace(&prefix);
|
|
|
|
|
|
|
|
// Don't really worry about error checking the the package/prefix for
|
|
|
|
|
|
|
|
// being valid. Assume the file is validated when it is created/edited.
|
|
|
|
|
|
|
|
(*prefix_map_)[package.ToString()] = prefix.ToString();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool LoadExpectedPackagePrefixes(map<string, string>* prefix_map,
|
|
|
|
|
|
|
|
string* out_expect_file_path,
|
|
|
|
|
|
|
|
string* out_error) {
|
|
|
|
|
|
|
|
const char* file_path = getenv("GPB_OBJC_EXPECTED_PACKAGE_PREFIXES");
|
|
|
|
|
|
|
|
if (file_path == NULL) {
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
|
|
|
fd = open(file_path, O_RDONLY);
|
|
|
|
|
|
|
|
} while (fd < 0 && errno == EINTR);
|
|
|
|
|
|
|
|
if (fd < 0) {
|
|
|
|
|
|
|
|
*out_error =
|
|
|
|
|
|
|
|
string(file_path) + ":0:0: error: Unable to open." + strerror(errno);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
io::FileInputStream file_stream(fd);
|
|
|
|
|
|
|
|
file_stream.SetCloseOnDelete(true);
|
|
|
|
|
|
|
|
*out_expect_file_path = file_path;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Parser parser(prefix_map);
|
|
|
|
|
|
|
|
const void* buf;
|
|
|
|
|
|
|
|
int buf_len;
|
|
|
|
|
|
|
|
while (file_stream.Next(&buf, &buf_len)) {
|
|
|
|
|
|
|
|
if (buf_len == 0) {
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!parser.ParseChunk(StringPiece(static_cast<const char*>(buf), buf_len))) {
|
|
|
|
|
|
|
|
*out_error = string(file_path) + ":" + SimpleItoa(parser.last_line()) +
|
|
|
|
|
|
|
|
":0: error: " + parser.error_str();
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return parser.Finish();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool ValidateObjCClassPrefix(const FileDescriptor* file, string* out_error) {
|
|
|
|
|
|
|
|
const string prefix = file->options().objc_class_prefix();
|
|
|
|
|
|
|
|
const string package = file->package();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
|
|
|
|
|
|
|
|
// error cases, so it seems to be ok to use as a back door for warnings.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// First Check: Warning - if there is a prefix, ensure it is is a reasonable
|
|
|
|
|
|
|
|
// value according to Apple's rules.
|
|
|
|
|
|
|
|
if (prefix.length()) {
|
|
|
|
|
|
|
|
if (!ascii_isupper(prefix[0])) {
|
|
|
|
|
|
|
|
cerr << endl
|
|
|
|
|
|
|
|
<< "protoc:0: warning: Invalid 'option objc_class_prefix = \""
|
|
|
|
|
|
|
|
<< prefix << "\";' in '" << file->name() << "';"
|
|
|
|
|
|
|
|
<< " it should start with a capital letter." << endl;
|
|
|
|
|
|
|
|
cerr.flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prefix.length() < 3) {
|
|
|
|
|
|
|
|
cerr << endl
|
|
|
|
|
|
|
|
<< "protoc:0: warning: Invalid 'option objc_class_prefix = \""
|
|
|
|
|
|
|
|
<< prefix << "\";' in '" << file->name() << "';"
|
|
|
|
|
|
|
|
<< " Apple recommends they should be at least 3 characters long."
|
|
|
|
|
|
|
|
<< endl;
|
|
|
|
|
|
|
|
cerr.flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Load any expected package prefixes to validate against those.
|
|
|
|
|
|
|
|
map<string, string> expected_package_prefixes;
|
|
|
|
|
|
|
|
string expect_file_path;
|
|
|
|
|
|
|
|
if (!LoadExpectedPackagePrefixes(&expected_package_prefixes,
|
|
|
|
|
|
|
|
&expect_file_path, out_error)) {
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// If there are no expected prefixes, out of here.
|
|
|
|
|
|
|
|
if (expected_package_prefixes.size() == 0) {
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Second Check: Error - See if there was an expected prefix for the package
|
|
|
|
|
|
|
|
// and report if it doesn't match.
|
|
|
|
|
|
|
|
map<string, string>::iterator package_match =
|
|
|
|
|
|
|
|
expected_package_prefixes.find(package);
|
|
|
|
|
|
|
|
if (package_match != expected_package_prefixes.end()) {
|
|
|
|
|
|
|
|
// There was an entry, and...
|
|
|
|
|
|
|
|
if (package_match->second == prefix) {
|
|
|
|
|
|
|
|
// ...it matches. All good, out of here!
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// ...it didn't match!
|
|
|
|
|
|
|
|
*out_error = "protoc:0: error: Expected 'option objc_class_prefix = \"" +
|
|
|
|
|
|
|
|
package_match->second + "\";' in '" + file->name() + "'";
|
|
|
|
|
|
|
|
if (prefix.length()) {
|
|
|
|
|
|
|
|
*out_error += "; but found '" + prefix + "' instead";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_error += ".";
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Third Check: Error - If there was a prefix make sure it wasn't expected
|
|
|
|
|
|
|
|
// for a different package instead (overlap is allowed, but it has to be
|
|
|
|
|
|
|
|
// listed as an expected overlap).
|
|
|
|
|
|
|
|
if (prefix.length()) {
|
|
|
|
|
|
|
|
for (map<string, string>::iterator i = expected_package_prefixes.begin();
|
|
|
|
|
|
|
|
i != expected_package_prefixes.end(); ++i) {
|
|
|
|
|
|
|
|
if (i->second == prefix) {
|
|
|
|
|
|
|
|
*out_error =
|
|
|
|
|
|
|
|
"protoc:0: error: Found 'option objc_class_prefix = \"" + prefix +
|
|
|
|
|
|
|
|
"\";' in '" + file->name() +
|
|
|
|
|
|
|
|
"'; that prefix is already used for 'package " + i->first +
|
|
|
|
|
|
|
|
";'. It can only be reused by listing it in the expected file (" +
|
|
|
|
|
|
|
|
expect_file_path + ").";
|
|
|
|
|
|
|
|
return false; // Only report first usage of the prefix.
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Fourth Check: Warning - If there was a prefix, and it wasn't expected,
|
|
|
|
|
|
|
|
// issue a warning suggesting it gets added to the file.
|
|
|
|
|
|
|
|
if (prefix.length()) {
|
|
|
|
|
|
|
|
cerr << endl
|
|
|
|
|
|
|
|
<< "protoc:0: warning: Found 'option objc_class_prefix = \"" << prefix
|
|
|
|
|
|
|
|
<< "\";' in '" << file->name() << "';"
|
|
|
|
|
|
|
|
<< " should you add it to the expected prefixes file ("
|
|
|
|
|
|
|
|
<< expect_file_path << ")?" << endl;
|
|
|
|
|
|
|
|
cerr.flush();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TextFormatDecodeData::AddString(int32 key,
|
|
|
|
void TextFormatDecodeData::AddString(int32 key,
|
|
|
|
const string& input_for_decode,
|
|
|
|
const string& input_for_decode,
|
|
|
|
const string& desired_output) {
|
|
|
|
const string& desired_output) {
|
|
|
@ -898,7 +1090,7 @@ class DecodeDataBuilder {
|
|
|
|
|
|
|
|
|
|
|
|
void AddChar(const char desired) {
|
|
|
|
void AddChar(const char desired) {
|
|
|
|
++segment_len_;
|
|
|
|
++segment_len_;
|
|
|
|
is_all_upper_ &= IsUpper(desired);
|
|
|
|
is_all_upper_ &= ascii_isupper(desired);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Push() {
|
|
|
|
void Push() {
|
|
|
@ -913,9 +1105,9 @@ class DecodeDataBuilder {
|
|
|
|
bool AddFirst(const char desired, const char input) {
|
|
|
|
bool AddFirst(const char desired, const char input) {
|
|
|
|
if (desired == input) {
|
|
|
|
if (desired == input) {
|
|
|
|
op_ = kOpAsIs;
|
|
|
|
op_ = kOpAsIs;
|
|
|
|
} else if (desired == ToUpper(input)) {
|
|
|
|
} else if (desired == ascii_toupper(input)) {
|
|
|
|
op_ = kOpFirstUpper;
|
|
|
|
op_ = kOpFirstUpper;
|
|
|
|
} else if (desired == ToLower(input)) {
|
|
|
|
} else if (desired == ascii_tolower(input)) {
|
|
|
|
op_ = kOpFirstLower;
|
|
|
|
op_ = kOpFirstLower;
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
// Can't be transformed to match.
|
|
|
|
// Can't be transformed to match.
|
|
|
@ -953,7 +1145,7 @@ bool DecodeDataBuilder::AddCharacter(const char desired, const char input) {
|
|
|
|
if (desired == input) {
|
|
|
|
if (desired == input) {
|
|
|
|
// If we aren't transforming it, or we're upper casing it and it is
|
|
|
|
// If we aren't transforming it, or we're upper casing it and it is
|
|
|
|
// supposed to be uppercase; just add it to the segment.
|
|
|
|
// supposed to be uppercase; just add it to the segment.
|
|
|
|
if ((op_ != kOpAllUpper) || IsUpper(desired)) {
|
|
|
|
if ((op_ != kOpAllUpper) || ascii_isupper(desired)) {
|
|
|
|
AddChar(desired);
|
|
|
|
AddChar(desired);
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -965,7 +1157,7 @@ bool DecodeDataBuilder::AddCharacter(const char desired, const char input) {
|
|
|
|
|
|
|
|
|
|
|
|
// If we need to uppercase, and everything so far has been uppercase,
|
|
|
|
// If we need to uppercase, and everything so far has been uppercase,
|
|
|
|
// promote op to AllUpper.
|
|
|
|
// promote op to AllUpper.
|
|
|
|
if ((desired == ToUpper(input)) && is_all_upper_) {
|
|
|
|
if ((desired == ascii_toupper(input)) && is_all_upper_) {
|
|
|
|
op_ = kOpAllUpper;
|
|
|
|
op_ = kOpAllUpper;
|
|
|
|
AddChar(desired);
|
|
|
|
AddChar(desired);
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|