| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 | // Protocol Buffers - Google's data interchange format// Copyright 2015 Google Inc.  All rights reserved.// https://developers.google.com/protocol-buffers///// Redistribution and use in source and binary forms, with or without// modification, are permitted provided that the following conditions are// met:////     * Redistributions of source code must retain the above copyright// notice, this list of conditions and the following disclaimer.//     * Redistributions in binary form must reproduce the above// copyright notice, this list of conditions and the following disclaimer// in the documentation and/or other materials provided with the// distribution.//     * Neither the name of Google Inc. nor the names of its// contributors may be used to endorse or promote products derived from// this software without specific prior written permission.//// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#import <Foundation/Foundation.h>#import "Conformance.pbobjc.h"#import "google/protobuf/TestMessagesProto2.pbobjc.h"#import "google/protobuf/TestMessagesProto3.pbobjc.h"static void Die(NSString *format, ...) __dead2;static BOOL verbose = NO;static int32_t testCount = 0;static void Die(NSString *format, ...) {  va_list args;  va_start(args, format);  NSString *msg = [[NSString alloc] initWithFormat:format arguments:args];  NSLog(@"%@", msg);  va_end(args);  [msg release];  exit(66);}static NSData *CheckedReadDataOfLength(NSFileHandle *handle, NSUInteger numBytes) {  NSData *data = [handle readDataOfLength:numBytes];  NSUInteger dataLen = data.length;  if (dataLen == 0) {    return nil;  // EOF.  }  if (dataLen != numBytes) {    Die(@"Failed to read the request length (%d), only got: %@",        numBytes, data);  }  return data;}static ConformanceResponse *DoTest(ConformanceRequest *request) {  ConformanceResponse *response = [ConformanceResponse message];  GPBMessage *testMessage = nil;  switch (request.payloadOneOfCase) {    case ConformanceRequest_Payload_OneOfCase_GPBUnsetOneOfCase:      response.runtimeError =          [NSString stringWithFormat:@"Request didn't have a payload: %@", request];      break;    case ConformanceRequest_Payload_OneOfCase_ProtobufPayload: {      Class msgClass = nil;      if ([request.messageType isEqual:@"protobuf_test_messages.proto3.TestAllTypesProto3"]) {        msgClass = [Proto3TestAllTypesProto3 class];      } else if ([request.messageType isEqual:@"protobuf_test_messages.proto2.TestAllTypesProto2"]) {        msgClass = [TestAllTypesProto2 class];      } else {        response.runtimeError =            [NSString stringWithFormat:@"Protobuf request had an unknown message_type: %@",                                       request.messageType];        break;      }      NSError *error = nil;      testMessage = [msgClass parseFromData:request.protobufPayload error:&error];      if (!testMessage) {        response.parseError =            [NSString stringWithFormat:@"Parse error: %@", error];      }      break;    }    case ConformanceRequest_Payload_OneOfCase_JsonPayload:      response.skipped = @"ObjC doesn't support parsing JSON";      break;    case ConformanceRequest_Payload_OneOfCase_JspbPayload:      response.skipped =          @"ConformanceRequest had a jspb_payload ConformanceRequest.payload;"          " those aren't supposed to happen with opensource.";      break;    case ConformanceRequest_Payload_OneOfCase_TextPayload:      response.skipped = @"ObjC doesn't support parsing TextFormat";      break;  }  if (testMessage) {    switch (request.requestedOutputFormat) {      case WireFormat_GPBUnrecognizedEnumeratorValue:      case WireFormat_Unspecified:        response.runtimeError =            [NSString stringWithFormat:@"Unrecognized/unspecified output format: %@", request];        break;      case WireFormat_Protobuf:        response.protobufPayload = testMessage.data;        if (!response.protobufPayload) {          response.serializeError =            [NSString stringWithFormat:@"Failed to make data from: %@", testMessage];        }        break;      case WireFormat_Json:        response.skipped = @"ObjC doesn't support generating JSON";        break;      case WireFormat_Jspb:        response.skipped =            @"ConformanceRequest had a requested_output_format of JSPB WireFormat; that"            " isn't supposed to happen with opensource.";        break;      case WireFormat_TextFormat:        // ObjC only has partial objc generation, so don't attempt any tests that need        // support.        response.skipped = @"ObjC doesn't support generating TextFormat";        break;    }  }  return response;}static uint32_t UInt32FromLittleEndianData(NSData *data) {  if (data.length != sizeof(uint32_t)) {    Die(@"Data not the right size for uint32_t: %@", data);  }  uint32_t value;  memcpy(&value, data.bytes, sizeof(uint32_t));  return CFSwapInt32LittleToHost(value);}static NSData *UInt32ToLittleEndianData(uint32_t num) {  uint32_t value = CFSwapInt32HostToLittle(num);  return [NSData dataWithBytes:&value length:sizeof(uint32_t)];}static BOOL DoTestIo(NSFileHandle *input, NSFileHandle *output) {  // See conformance_test_runner.cc for the wire format.  NSData *data = CheckedReadDataOfLength(input, sizeof(uint32_t));  if (!data) {    // EOF.    return NO;  }  uint32_t numBytes = UInt32FromLittleEndianData(data);  data = CheckedReadDataOfLength(input, numBytes);  if (!data) {    Die(@"Failed to read request");  }  NSError *error = nil;  ConformanceRequest *request = [ConformanceRequest parseFromData:data                                                            error:&error];  if (!request) {    Die(@"Failed to parse the message data: %@", error);  }  ConformanceResponse *response = DoTest(request);  if (!response) {    Die(@"Failed to make a reply from %@", request);  }  data = response.data;  [output writeData:UInt32ToLittleEndianData((int32_t)data.length)];  [output writeData:data];  if (verbose) {    NSLog(@"Request: %@", request);    NSLog(@"Response: %@", response);  }  ++testCount;  return YES;}int main(int argc, const char *argv[]) {  @autoreleasepool {    NSFileHandle *input = [[NSFileHandle fileHandleWithStandardInput] retain];    NSFileHandle *output = [[NSFileHandle fileHandleWithStandardOutput] retain];    BOOL notDone = YES;    while (notDone) {      @autoreleasepool {        notDone = DoTestIo(input, output);      }    }    NSLog(@"Received EOF from test runner after %d tests, exiting.", testCount);  }  return 0;}
 |