|
|
@ -9,7 +9,9 @@ |
|
|
|
#include <unistd.h> |
|
|
|
#include <unistd.h> |
|
|
|
|
|
|
|
|
|
|
|
#include "conformance/conformance.upb.h" |
|
|
|
#include "conformance/conformance.upb.h" |
|
|
|
#include "src/google/protobuf/test_messages_proto3.upb.h" |
|
|
|
#include "src/google/protobuf/test_messages_proto2.upbdefs.h" |
|
|
|
|
|
|
|
#include "src/google/protobuf/test_messages_proto3.upbdefs.h" |
|
|
|
|
|
|
|
#include "upb/reflection.h" |
|
|
|
|
|
|
|
|
|
|
|
int test_count = 0; |
|
|
|
int test_count = 0; |
|
|
|
|
|
|
|
|
|
|
@ -39,138 +41,144 @@ void CheckedWrite(int fd, const void *buf, size_t len) { |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
bool strview_eql(upb_strview view, const char *str) { |
|
|
|
typedef struct { |
|
|
|
return view.size == strlen(str) && memcmp(view.data, str, view.size) == 0; |
|
|
|
const conformance_ConformanceRequest *request; |
|
|
|
} |
|
|
|
conformance_ConformanceResponse *response; |
|
|
|
|
|
|
|
upb_arena *arena; |
|
|
|
static const char *proto3_msg = |
|
|
|
const upb_symtab *symtab; |
|
|
|
"protobuf_test_messages.proto3.TestAllTypesProto3"; |
|
|
|
} ctx; |
|
|
|
|
|
|
|
|
|
|
|
void DoTest( |
|
|
|
|
|
|
|
const conformance_ConformanceRequest* request, |
|
|
|
|
|
|
|
conformance_ConformanceResponse *response, |
|
|
|
|
|
|
|
upb_arena *arena) { |
|
|
|
|
|
|
|
protobuf_test_messages_proto3_TestAllTypesProto3 *test_message; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!strview_eql(conformance_ConformanceRequest_message_type(request), |
|
|
|
|
|
|
|
proto3_msg)) { |
|
|
|
|
|
|
|
static const char msg[] = "Only proto3 for now."; |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
|
|
|
|
response, upb_strview_make(msg, sizeof(msg))); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
switch (conformance_ConformanceRequest_payload_case(request)) { |
|
|
|
|
|
|
|
case conformance_ConformanceRequest_payload_protobuf_payload: { |
|
|
|
|
|
|
|
upb_strview payload = conformance_ConformanceRequest_protobuf_payload(request); |
|
|
|
|
|
|
|
test_message = protobuf_test_messages_proto3_TestAllTypesProto3_parse( |
|
|
|
|
|
|
|
payload.data, payload.size, arena); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!test_message) { |
|
|
|
bool parse_proto(upb_msg *msg, const upb_msgdef *m, const ctx* c) { |
|
|
|
|
|
|
|
upb_strview proto = |
|
|
|
|
|
|
|
conformance_ConformanceRequest_protobuf_payload(c->request); |
|
|
|
|
|
|
|
if (upb_decode(proto.data, proto.size, msg, upb_msgdef_layout(m), c->arena)) { |
|
|
|
|
|
|
|
return true; |
|
|
|
|
|
|
|
} else { |
|
|
|
static const char msg[] = "Parse error"; |
|
|
|
static const char msg[] = "Parse error"; |
|
|
|
conformance_ConformanceResponse_set_parse_error( |
|
|
|
conformance_ConformanceResponse_set_parse_error( |
|
|
|
response, upb_strview_make(msg, sizeof(msg))); |
|
|
|
c->response, upb_strview_make(msg, strlen(msg))); |
|
|
|
return; |
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void serialize_proto(const upb_msg *msg, const upb_msgdef *m, const ctx *c) { |
|
|
|
|
|
|
|
size_t len; |
|
|
|
|
|
|
|
char *data = upb_encode(msg, upb_msgdef_layout(m), c->arena, &len); |
|
|
|
|
|
|
|
if (data) { |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_protobuf_payload( |
|
|
|
|
|
|
|
c->response, upb_strview_make(data, len)); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
static const char msg[] = "Error serializing."; |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_serialize_error( |
|
|
|
|
|
|
|
c->response, upb_strview_make(msg, strlen(msg))); |
|
|
|
} |
|
|
|
} |
|
|
|
break; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool parse_input(upb_msg *msg, const upb_msgdef *m, const ctx* c) { |
|
|
|
|
|
|
|
switch (conformance_ConformanceRequest_payload_case(c->request)) { |
|
|
|
|
|
|
|
case conformance_ConformanceRequest_payload_protobuf_payload: |
|
|
|
|
|
|
|
return parse_proto(msg, m, c); |
|
|
|
case conformance_ConformanceRequest_payload_NOT_SET: |
|
|
|
case conformance_ConformanceRequest_payload_NOT_SET: |
|
|
|
fprintf(stderr, "conformance_upb: Request didn't have payload.\n"); |
|
|
|
fprintf(stderr, "conformance_upb: Request didn't have payload.\n"); |
|
|
|
return; |
|
|
|
return false; |
|
|
|
|
|
|
|
|
|
|
|
default: { |
|
|
|
default: { |
|
|
|
static const char msg[] = "Unsupported input format."; |
|
|
|
static const char msg[] = "Unsupported input format."; |
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
response, upb_strview_make(msg, sizeof(msg))); |
|
|
|
c->response, upb_strview_make(msg, strlen(msg))); |
|
|
|
return; |
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
switch (conformance_ConformanceRequest_requested_output_format(request)) { |
|
|
|
void write_output(const upb_msg *msg, const upb_msgdef *m, const ctx* c) { |
|
|
|
|
|
|
|
switch (conformance_ConformanceRequest_requested_output_format(c->request)) { |
|
|
|
case conformance_UNSPECIFIED: |
|
|
|
case conformance_UNSPECIFIED: |
|
|
|
fprintf(stderr, "conformance_upb: Unspecified output format.\n"); |
|
|
|
fprintf(stderr, "conformance_upb: Unspecified output format.\n"); |
|
|
|
exit(1); |
|
|
|
exit(1); |
|
|
|
|
|
|
|
case conformance_PROTOBUF: |
|
|
|
case conformance_PROTOBUF: { |
|
|
|
serialize_proto(msg, m, c); |
|
|
|
size_t serialized_len; |
|
|
|
|
|
|
|
char *serialized = |
|
|
|
|
|
|
|
protobuf_test_messages_proto3_TestAllTypesProto3_serialize( |
|
|
|
|
|
|
|
test_message, arena, &serialized_len); |
|
|
|
|
|
|
|
if (!serialized) { |
|
|
|
|
|
|
|
static const char msg[] = "Error serializing."; |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_serialize_error( |
|
|
|
|
|
|
|
response, upb_strview_make(msg, sizeof(msg))); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_protobuf_payload( |
|
|
|
|
|
|
|
response, upb_strview_make(serialized, serialized_len)); |
|
|
|
|
|
|
|
break; |
|
|
|
break; |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default: { |
|
|
|
default: { |
|
|
|
static const char msg[] = "Unsupported output format."; |
|
|
|
static const char msg[] = "Unsupported output format."; |
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
response, upb_strview_make(msg, sizeof(msg))); |
|
|
|
c->response, upb_strview_make(msg, strlen(msg))); |
|
|
|
return; |
|
|
|
break; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void DoTest(const ctx* c) { |
|
|
|
|
|
|
|
upb_msg *msg; |
|
|
|
|
|
|
|
upb_strview name = conformance_ConformanceRequest_message_type(c->request); |
|
|
|
|
|
|
|
const upb_msgdef *m = upb_symtab_lookupmsg2(c->symtab, name.data, name.size); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!m) { |
|
|
|
|
|
|
|
static const char msg[] = "Unknown message type."; |
|
|
|
|
|
|
|
conformance_ConformanceResponse_set_skipped( |
|
|
|
|
|
|
|
c->response, upb_strview_make(msg, strlen(msg))); |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
bool DoTestIo(void) { |
|
|
|
msg = upb_msg_new(m, c->arena); |
|
|
|
upb_arena *arena; |
|
|
|
|
|
|
|
upb_alloc *alloc; |
|
|
|
if (parse_input(msg, m, c)) { |
|
|
|
|
|
|
|
write_output(msg, m, c); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool DoTestIo(const upb_symtab *symtab) { |
|
|
|
upb_status status; |
|
|
|
upb_status status; |
|
|
|
char *serialized_input; |
|
|
|
char *input; |
|
|
|
char *serialized_output; |
|
|
|
char *output; |
|
|
|
uint32_t input_size; |
|
|
|
uint32_t input_size; |
|
|
|
size_t output_size; |
|
|
|
size_t output_size; |
|
|
|
conformance_ConformanceRequest *request; |
|
|
|
ctx c; |
|
|
|
conformance_ConformanceResponse *response; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!CheckedRead(STDIN_FILENO, &input_size, sizeof(uint32_t))) { |
|
|
|
if (!CheckedRead(STDIN_FILENO, &input_size, sizeof(uint32_t))) { |
|
|
|
/* EOF. */ |
|
|
|
/* EOF. */ |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
arena = upb_arena_new(); |
|
|
|
c.symtab = symtab; |
|
|
|
alloc = upb_arena_alloc(arena); |
|
|
|
c.arena = upb_arena_new(); |
|
|
|
serialized_input = upb_malloc(alloc, input_size); |
|
|
|
input = upb_arena_malloc(c.arena, input_size); |
|
|
|
|
|
|
|
|
|
|
|
if (!CheckedRead(STDIN_FILENO, serialized_input, input_size)) { |
|
|
|
if (!CheckedRead(STDIN_FILENO, input, input_size)) { |
|
|
|
fprintf(stderr, "conformance_upb: unexpected EOF on stdin.\n"); |
|
|
|
fprintf(stderr, "conformance_upb: unexpected EOF on stdin.\n"); |
|
|
|
exit(1); |
|
|
|
exit(1); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
request = |
|
|
|
c.request = conformance_ConformanceRequest_parse(input, input_size, c.arena); |
|
|
|
conformance_ConformanceRequest_parse(serialized_input, input_size, arena); |
|
|
|
c.response = conformance_ConformanceResponse_new(c.arena); |
|
|
|
response = conformance_ConformanceResponse_new(arena); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (request) { |
|
|
|
if (c.request) { |
|
|
|
DoTest(request, response, arena); |
|
|
|
DoTest(&c); |
|
|
|
} else { |
|
|
|
} else { |
|
|
|
fprintf(stderr, "conformance_upb: parse of ConformanceRequest failed: %s\n", |
|
|
|
fprintf(stderr, "conformance_upb: parse of ConformanceRequest failed: %s\n", |
|
|
|
upb_status_errmsg(&status)); |
|
|
|
upb_status_errmsg(&status)); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
serialized_output = conformance_ConformanceResponse_serialize( |
|
|
|
output = conformance_ConformanceResponse_serialize(c.response, c.arena, |
|
|
|
response, arena, &output_size); |
|
|
|
&output_size); |
|
|
|
|
|
|
|
|
|
|
|
CheckedWrite(STDOUT_FILENO, &output_size, sizeof(uint32_t)); |
|
|
|
CheckedWrite(STDOUT_FILENO, &output_size, sizeof(uint32_t)); |
|
|
|
CheckedWrite(STDOUT_FILENO, serialized_output, output_size); |
|
|
|
CheckedWrite(STDOUT_FILENO, output, output_size); |
|
|
|
|
|
|
|
|
|
|
|
test_count++; |
|
|
|
test_count++; |
|
|
|
|
|
|
|
|
|
|
|
upb_arena_free(arena); |
|
|
|
upb_arena_free(c.arena); |
|
|
|
|
|
|
|
|
|
|
|
return true; |
|
|
|
return true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
int main(void) { |
|
|
|
int main(void) { |
|
|
|
|
|
|
|
upb_symtab *symtab = upb_symtab_new(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
protobuf_test_messages_proto2_TestAllTypesProto2_getmsgdef(symtab); |
|
|
|
|
|
|
|
protobuf_test_messages_proto3_TestAllTypesProto3_getmsgdef(symtab); |
|
|
|
|
|
|
|
|
|
|
|
while (1) { |
|
|
|
while (1) { |
|
|
|
if (!DoTestIo()) { |
|
|
|
if (!DoTestIo(symtab)) { |
|
|
|
fprintf(stderr, "conformance_upb: received EOF from test runner " |
|
|
|
fprintf(stderr, "conformance_upb: received EOF from test runner " |
|
|
|
"after %d tests, exiting\n", test_count); |
|
|
|
"after %d tests, exiting\n", test_count); |
|
|
|
return 0; |
|
|
|
return 0; |
|
|
|