mirror of
https://github.com/valitydev/thrift.git
synced 2024-11-06 10:25:18 +00:00
THRIFT-2905 Cocoa compiler should have option to produce "modern" Objective-C
Client: Cocoa (ObjectiveC & Swift) Author: Kevin Wooten <kevin@wooten.com> This closes #539
This commit is contained in:
parent
983bf7de41
commit
56e5b9b01b
18
Thrift.podspec
Normal file
18
Thrift.podspec
Normal file
@ -0,0 +1,18 @@
|
||||
Pod::Spec.new do |s|
|
||||
s.name = "Thrift"
|
||||
s.version = "1.0.0"
|
||||
s.summary = "Apache Thrift is a lightweight, language-independent software stack with an associated code generation mechanism for RPC."
|
||||
s.description = <<-DESC
|
||||
The Apache Thrift software framework, for scalable cross-language services development, combines a software stack with a code generation engine to build services that work efficiently and seamlessly between C++, Java, Python, PHP, Ruby, Erlang, Perl, Haskell, C#, Cocoa, JavaScript, Node.js, Smalltalk, OCaml and Delphi and other languages.
|
||||
DESC
|
||||
s.homepage = "http://thrift.apache.org"
|
||||
s.license = { :type => 'Apache License, Version 2.0', :url => 'https://raw.github.com/apache/thrift/thrift-0.9.0/LICENSE' }
|
||||
s.author = { "The Apache Software Foundation" => "apache@apache.org" }
|
||||
s.requires_arc = true
|
||||
s.ios.deployment_target = '7.0'
|
||||
s.osx.deployment_target = '10.8'
|
||||
s.ios.framework = 'CFNetwork'
|
||||
s.osx.framework = 'CoreServices'
|
||||
s.source = { :git => "https://github.com/apache/thrift.git", :tag => "thrift-1.0.0" }
|
||||
s.source_files = 'lib/cocoa/src/**/*.{h,m,swift}'
|
||||
end
|
@ -109,6 +109,7 @@ THRIFT_ADD_COMPILER(perl "Enable compiler for Perl" ON)
|
||||
THRIFT_ADD_COMPILER(php "Enable compiler for PHP" ON)
|
||||
THRIFT_ADD_COMPILER(erl "Enable compiler for Erlang" ON)
|
||||
THRIFT_ADD_COMPILER(cocoa "Enable compiler for Cocoa Objective-C" ON)
|
||||
THRIFT_ADD_COMPILER(swift "Enable compiler for Cocoa Swift" ON)
|
||||
THRIFT_ADD_COMPILER(st "Enable compiler for Smalltalk" ON)
|
||||
THRIFT_ADD_COMPILER(ocaml "Enable compiler for OCaml" ON)
|
||||
THRIFT_ADD_COMPILER(hs "Enable compiler for Haskell" ON)
|
||||
|
@ -83,6 +83,7 @@ thrift_SOURCES += src/generate/t_c_glib_generator.cc \
|
||||
src/generate/t_php_generator.cc \
|
||||
src/generate/t_erl_generator.cc \
|
||||
src/generate/t_cocoa_generator.cc \
|
||||
src/generate/t_swift_generator.cc \
|
||||
src/generate/t_st_generator.cc \
|
||||
src/generate/t_ocaml_generator.cc \
|
||||
src/generate/t_hs_generator.cc \
|
||||
|
@ -78,6 +78,7 @@
|
||||
<ClCompile Include="src\generate\t_py_generator.cc" />
|
||||
<ClCompile Include="src\generate\t_rb_generator.cc" />
|
||||
<ClCompile Include="src\generate\t_st_generator.cc" />
|
||||
<ClCompile Include="src\generate\t_swift_generator.cc" />
|
||||
<ClCompile Include="src\generate\t_xsd_generator.cc" />
|
||||
<ClCompile Include="src\main.cc" />
|
||||
<ClCompile Include="src\md5.c" />
|
||||
|
@ -165,6 +165,9 @@
|
||||
<ClCompile Include="src\generate\t_st_generator.cc">
|
||||
<Filter>generate</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="src\generate\t_swift_generator.cc">
|
||||
<Filter>generate</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="src\generate\t_xsd_generator.cc">
|
||||
<Filter>generate</Filter>
|
||||
</ClCompile>
|
||||
|
File diff suppressed because it is too large
Load Diff
2145
compiler/cpp/src/generate/t_swift_generator.cc
Normal file
2145
compiler/cpp/src/generate/t_swift_generator.cc
Normal file
File diff suppressed because it is too large
Load Diff
55
lib/cocoa/src/TApplicationError.h
Normal file
55
lib/cocoa/src/TApplicationError.h
Normal file
@ -0,0 +1,55 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TProtocol.h"
|
||||
|
||||
extern NSString *TApplicationErrorDomain;
|
||||
|
||||
typedef NS_ENUM (int, TApplicationError) {
|
||||
TApplicationErrorUnknown = 0,
|
||||
TApplicationErrorUnknownMethod = 1,
|
||||
TApplicationErrorInvalidMessageType = 2,
|
||||
TApplicationErrorWrongMethodName = 3,
|
||||
TApplicationErrorBadSequenceId = 4,
|
||||
TApplicationErrorMissingResult = 5,
|
||||
TApplicationErrorInternalError = 6,
|
||||
TApplicationErrorProtocolError = 7,
|
||||
TApplicationErrorInvalidTransform = 8,
|
||||
TApplicationErrorInvalidProtocol = 9,
|
||||
TApplicationErrorUnsupportedClientType = 10,
|
||||
};
|
||||
|
||||
|
||||
extern NSString *TApplicationErrorNameKey;
|
||||
extern NSString *TApplicationErrorReasonKey;
|
||||
extern NSString *TApplicationErrorMethodKey;
|
||||
|
||||
|
||||
@interface NSError (TApplicationError)
|
||||
|
||||
@property (readonly, copy) NSString *name;
|
||||
@property (readonly, copy) NSString *reason;
|
||||
|
||||
+(instancetype) errorWithType:(TApplicationError)type reason:(NSString *)reason;
|
||||
|
||||
+(instancetype) read:(id<TProtocol>)protocol;
|
||||
|
||||
-(BOOL) write:(id<TProtocol>)outProtocol error:(NSError *__autoreleasing *)error;
|
||||
|
||||
@end
|
231
lib/cocoa/src/TApplicationError.m
Normal file
231
lib/cocoa/src/TApplicationError.m
Normal file
@ -0,0 +1,231 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TApplicationError.h"
|
||||
#import "TProtocolUtil.h"
|
||||
|
||||
|
||||
NSString *TApplicationErrorDomain = @"TApplicationErrorDomain";
|
||||
|
||||
|
||||
NSString *TApplicationErrorNameKey = @"name";
|
||||
NSString *TApplicationErrorReasonKey = @"reason";
|
||||
NSString *TApplicationErrorMethodKey = @"method";
|
||||
|
||||
|
||||
@implementation NSError (TApplicationError)
|
||||
|
||||
-(NSString *) reason
|
||||
{
|
||||
return self.userInfo[TApplicationErrorReasonKey];
|
||||
}
|
||||
|
||||
-(NSString *) name
|
||||
{
|
||||
return self.userInfo[TApplicationErrorNameKey];
|
||||
}
|
||||
|
||||
+(instancetype) errorWithType:(TApplicationError)type reason:(NSString *)reason
|
||||
{
|
||||
NSString *name;
|
||||
switch (type) {
|
||||
case TApplicationErrorUnknownMethod:
|
||||
name = @"Unknown Method";
|
||||
break;
|
||||
|
||||
case TApplicationErrorInvalidMessageType:
|
||||
name = @"Invalid Message Type";
|
||||
break;
|
||||
|
||||
case TApplicationErrorWrongMethodName:
|
||||
name = @"Wrong Method Name";
|
||||
break;
|
||||
|
||||
case TApplicationErrorBadSequenceId:
|
||||
name = @"Bad Sequence ID";
|
||||
break;
|
||||
|
||||
case TApplicationErrorMissingResult:
|
||||
name = @"Missing Result";
|
||||
break;
|
||||
|
||||
case TApplicationErrorInternalError:
|
||||
name = @"Internal Error";
|
||||
break;
|
||||
|
||||
case TApplicationErrorProtocolError:
|
||||
name = @"Protocol Error";
|
||||
break;
|
||||
|
||||
case TApplicationErrorInvalidTransform:
|
||||
name = @"Invalid Transform";
|
||||
break;
|
||||
|
||||
case TApplicationErrorInvalidProtocol:
|
||||
name = @"Invalid Protocol";
|
||||
break;
|
||||
|
||||
case TApplicationErrorUnsupportedClientType:
|
||||
name = @"Unsupported Client Type";
|
||||
break;
|
||||
|
||||
default:
|
||||
name = @"Unknown";
|
||||
break;
|
||||
}
|
||||
|
||||
NSDictionary *userInfo;
|
||||
if (reason) {
|
||||
userInfo = @{TApplicationErrorNameKey:name,
|
||||
TApplicationErrorReasonKey:reason};
|
||||
}
|
||||
else {
|
||||
userInfo = @{TApplicationErrorNameKey:name};
|
||||
}
|
||||
|
||||
return [NSError errorWithDomain:TApplicationErrorDomain
|
||||
code:type
|
||||
userInfo:userInfo];
|
||||
}
|
||||
|
||||
|
||||
+(instancetype) read:(id<TProtocol>)protocol
|
||||
{
|
||||
NSString *reason = nil;
|
||||
SInt32 type = TApplicationErrorUnknown;
|
||||
SInt32 fieldType;
|
||||
SInt32 fieldID;
|
||||
|
||||
NSError *error;
|
||||
if (![protocol readStructBeginReturningName:NULL error:&error]) {
|
||||
return error;
|
||||
}
|
||||
|
||||
while (true) {
|
||||
|
||||
if (![protocol readFieldBeginReturningName:NULL
|
||||
type:&fieldType
|
||||
fieldID:&fieldID
|
||||
error:&error])
|
||||
{
|
||||
return error;
|
||||
}
|
||||
|
||||
if (fieldType == TTypeSTOP) {
|
||||
break;
|
||||
}
|
||||
|
||||
switch (fieldID) {
|
||||
case 1:
|
||||
if (fieldType == TTypeSTRING) {
|
||||
if (![protocol readString:&reason error:&error]) {
|
||||
return error;
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (![TProtocolUtil skipType:fieldType onProtocol:protocol error:&error]) {
|
||||
return error;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case 2:
|
||||
if (fieldType == TTypeI32) {
|
||||
if (![protocol readI32:&type error:&error]) {
|
||||
return error;
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (![TProtocolUtil skipType:fieldType onProtocol:protocol error:&error]) {
|
||||
return error;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
if (![TProtocolUtil skipType:fieldType onProtocol:protocol error:&error]) {
|
||||
return error;
|
||||
}
|
||||
break;
|
||||
}
|
||||
if (![protocol readFieldEnd:&error]) {
|
||||
return error;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (![protocol readStructEnd:&error]) {
|
||||
return error;
|
||||
}
|
||||
|
||||
return [NSError errorWithType:type reason:reason];
|
||||
}
|
||||
|
||||
|
||||
-(BOOL) write:(id<TProtocol>)protocol error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
if (![protocol writeStructBeginWithName:@"TApplicationException" error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (self.localizedDescription != nil) {
|
||||
if (![protocol writeFieldBeginWithName:@"message"
|
||||
type:TTypeSTRING
|
||||
fieldID:1 error:error])
|
||||
{
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeString:self.localizedDescription error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeFieldEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
|
||||
if (![protocol writeFieldBeginWithName:@"type"
|
||||
type:TTypeI32
|
||||
fieldID:2
|
||||
error:error])
|
||||
{
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeI32:(SInt32)self.code error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeFieldEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeFieldStop:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![protocol writeStructEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
@end
|
@ -1,49 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TException.h"
|
||||
#import "TProtocol.h"
|
||||
|
||||
enum {
|
||||
TApplicationException_UNKNOWN = 0,
|
||||
TApplicationException_UNKNOWN_METHOD = 1,
|
||||
TApplicationException_INVALID_MESSAGE_TYPE = 2,
|
||||
TApplicationException_WRONG_METHOD_NAME = 3,
|
||||
TApplicationException_BAD_SEQUENCE_ID = 4,
|
||||
TApplicationException_MISSING_RESULT = 5,
|
||||
TApplicationException_INTERNAL_ERROR = 6,
|
||||
TApplicationException_PROTOCOL_ERROR = 7,
|
||||
TApplicationException_INVALID_TRANSFORM = 8,
|
||||
TApplicationException_INVALID_PROTOCOL = 9,
|
||||
TApplicationException_UNSUPPORTED_CLIENT_TYPE = 10
|
||||
};
|
||||
|
||||
// FIXME
|
||||
@interface TApplicationException : TException {
|
||||
int mType;
|
||||
}
|
||||
|
||||
+ (TApplicationException *) read: (id <TProtocol>) protocol;
|
||||
|
||||
- (void) write: (id <TProtocol>) protocol;
|
||||
|
||||
+ (TApplicationException *) exceptionWithType: (int) type
|
||||
reason: (NSString *) message;
|
||||
|
||||
@end
|
@ -1,146 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TApplicationException.h"
|
||||
#import "TProtocolUtil.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
@implementation TApplicationException
|
||||
|
||||
- (id) initWithType: (int) type
|
||||
reason: (NSString *) reason
|
||||
{
|
||||
mType = type;
|
||||
|
||||
NSString * name;
|
||||
switch (type) {
|
||||
case TApplicationException_UNKNOWN_METHOD:
|
||||
name = @"Unknown method";
|
||||
break;
|
||||
case TApplicationException_INVALID_MESSAGE_TYPE:
|
||||
name = @"Invalid message type";
|
||||
break;
|
||||
case TApplicationException_WRONG_METHOD_NAME:
|
||||
name = @"Wrong method name";
|
||||
break;
|
||||
case TApplicationException_BAD_SEQUENCE_ID:
|
||||
name = @"Bad sequence ID";
|
||||
break;
|
||||
case TApplicationException_MISSING_RESULT:
|
||||
name = @"Missing result";
|
||||
break;
|
||||
case TApplicationException_INTERNAL_ERROR:
|
||||
name = @"Internal error";
|
||||
break;
|
||||
case TApplicationException_PROTOCOL_ERROR:
|
||||
name = @"Protocol error";
|
||||
break;
|
||||
case TApplicationException_INVALID_TRANSFORM:
|
||||
name = @"Invalid transform";
|
||||
break;
|
||||
case TApplicationException_INVALID_PROTOCOL:
|
||||
name = @"Invalid protocol";
|
||||
break;
|
||||
case TApplicationException_UNSUPPORTED_CLIENT_TYPE:
|
||||
name = @"Unsupported client type";
|
||||
break;
|
||||
default:
|
||||
name = @"Unknown";
|
||||
break;
|
||||
}
|
||||
|
||||
self = [super initWithName: name reason: reason userInfo: nil];
|
||||
return self;
|
||||
}
|
||||
|
||||
|
||||
+ (TApplicationException *) read: (id <TProtocol>) protocol
|
||||
{
|
||||
NSString * reason = nil;
|
||||
int type = TApplicationException_UNKNOWN;
|
||||
int fieldType;
|
||||
int fieldID;
|
||||
|
||||
[protocol readStructBeginReturningName: NULL];
|
||||
|
||||
while (true) {
|
||||
[protocol readFieldBeginReturningName: NULL
|
||||
type: &fieldType
|
||||
fieldID: &fieldID];
|
||||
if (fieldType == TType_STOP) {
|
||||
break;
|
||||
}
|
||||
switch (fieldID) {
|
||||
case 1:
|
||||
if (fieldType == TType_STRING) {
|
||||
reason = [protocol readString];
|
||||
} else {
|
||||
[TProtocolUtil skipType: fieldType onProtocol: protocol];
|
||||
}
|
||||
break;
|
||||
case 2:
|
||||
if (fieldType == TType_I32) {
|
||||
type = [protocol readI32];
|
||||
} else {
|
||||
[TProtocolUtil skipType: fieldType onProtocol: protocol];
|
||||
}
|
||||
break;
|
||||
default:
|
||||
[TProtocolUtil skipType: fieldType onProtocol: protocol];
|
||||
break;
|
||||
}
|
||||
[protocol readFieldEnd];
|
||||
}
|
||||
[protocol readStructEnd];
|
||||
|
||||
return [TApplicationException exceptionWithType: type reason: reason];
|
||||
}
|
||||
|
||||
|
||||
- (void) write: (id <TProtocol>) protocol
|
||||
{
|
||||
[protocol writeStructBeginWithName: @"TApplicationException"];
|
||||
|
||||
if ([self reason] != nil) {
|
||||
[protocol writeFieldBeginWithName: @"message"
|
||||
type: TType_STRING
|
||||
fieldID: 1];
|
||||
[protocol writeString: [self reason]];
|
||||
[protocol writeFieldEnd];
|
||||
}
|
||||
|
||||
[protocol writeFieldBeginWithName: @"type"
|
||||
type: TType_I32
|
||||
fieldID: 2];
|
||||
[protocol writeI32: mType];
|
||||
[protocol writeFieldEnd];
|
||||
|
||||
[protocol writeFieldStop];
|
||||
[protocol writeStructEnd];
|
||||
}
|
||||
|
||||
|
||||
+ (TApplicationException *) exceptionWithType: (int) type
|
||||
reason: (NSString *) reason
|
||||
{
|
||||
return [[[TApplicationException alloc] initWithType: type
|
||||
reason: reason] autorelease_stub];
|
||||
}
|
||||
|
||||
@end
|
@ -18,13 +18,10 @@
|
||||
*/
|
||||
|
||||
#import "TProtocol.h"
|
||||
#import "TApplicationException.h"
|
||||
#import "TApplicationError.h"
|
||||
|
||||
@interface TBaseClient : NSObject {
|
||||
id <TProtocol> inProtocol;
|
||||
id <TProtocol> outProtocol;
|
||||
}
|
||||
@interface TBaseClient : NSObject
|
||||
|
||||
- (TApplicationException *)checkIncomingMessageException;
|
||||
-(NSError *) checkIncomingMessageException:(id<TProtocol>)protocol;
|
||||
|
||||
@end
|
||||
|
@ -18,29 +18,34 @@
|
||||
*/
|
||||
|
||||
#import "TBaseClient.h"
|
||||
#import "TApplicationException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TApplicationError.h"
|
||||
|
||||
|
||||
@interface TBaseClient ()
|
||||
@end
|
||||
|
||||
|
||||
@implementation TBaseClient
|
||||
|
||||
- (void) dealloc
|
||||
-(NSError *) checkIncomingMessageException:(id<TProtocol>)inProtocol
|
||||
{
|
||||
[inProtocol release_stub];
|
||||
[outProtocol release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
NSError *thriftError;
|
||||
|
||||
- (TApplicationException *)checkIncomingMessageException
|
||||
{
|
||||
int msgType = 0;
|
||||
[inProtocol readMessageBeginReturningName: nil type: &msgType sequenceID: NULL];
|
||||
if (msgType == TMessageType_EXCEPTION) {
|
||||
TApplicationException * x = [TApplicationException read: inProtocol];
|
||||
[inProtocol readMessageEnd];
|
||||
return x;
|
||||
}
|
||||
|
||||
return nil;
|
||||
SInt32 msgType = 0;
|
||||
if (![inProtocol readMessageBeginReturningName:nil type:&msgType sequenceID:NULL error:&thriftError]) {
|
||||
return thriftError;
|
||||
}
|
||||
|
||||
if (msgType == TMessageTypeEXCEPTION) {
|
||||
|
||||
thriftError = [NSError read:inProtocol];
|
||||
|
||||
[inProtocol readMessageEnd:NULL];
|
||||
|
||||
return thriftError;
|
||||
}
|
||||
|
||||
return nil;
|
||||
}
|
||||
|
||||
@end
|
||||
|
122
lib/cocoa/src/TBinary.swift
Normal file
122
lib/cocoa/src/TBinary.swift
Normal file
@ -0,0 +1,122 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public struct TBinary : TSerializable {
|
||||
|
||||
public static var thriftType : TType { return .STRING }
|
||||
|
||||
private var storage : NSData
|
||||
|
||||
public init() {
|
||||
self.storage = NSData()
|
||||
}
|
||||
|
||||
public init(contentsOfFile file: String, options: NSDataReadingOptions = []) throws {
|
||||
self.storage = try NSData(contentsOfFile: file, options: options)
|
||||
}
|
||||
|
||||
public init(contentsOfURL URL: NSURL, options: NSDataReadingOptions = []) throws {
|
||||
self.storage = try NSData(contentsOfURL: URL, options: options)
|
||||
}
|
||||
|
||||
public init?(base64EncodedData base64Data: NSData, options: NSDataBase64DecodingOptions = []) {
|
||||
guard let storage = NSData(base64EncodedData: base64Data, options: options) else {
|
||||
return nil
|
||||
}
|
||||
self.storage = storage
|
||||
}
|
||||
|
||||
public init(data: NSData) {
|
||||
self.storage = data
|
||||
}
|
||||
|
||||
public var length : Int {
|
||||
return storage.length
|
||||
}
|
||||
|
||||
public var hashValue : Int {
|
||||
return storage.hashValue
|
||||
}
|
||||
|
||||
public var bytes : UnsafePointer<Void> {
|
||||
return storage.bytes
|
||||
}
|
||||
|
||||
public func getBytes(buffer: UnsafeMutablePointer<Void>, length: Int) {
|
||||
storage.getBytes(buffer, length: length)
|
||||
}
|
||||
|
||||
public func getBytes(buffer: UnsafeMutablePointer<Void>, range: Range<Int>) {
|
||||
storage.getBytes(buffer, range: NSRange(range))
|
||||
}
|
||||
|
||||
public func subBinaryWithRange(range: Range<Int>) -> TBinary {
|
||||
return TBinary(data: storage.subdataWithRange(NSRange(range)))
|
||||
}
|
||||
|
||||
public func writeToFile(path: String, options: NSDataWritingOptions = []) throws {
|
||||
try storage.writeToFile(path, options: options)
|
||||
}
|
||||
|
||||
public func writeToURL(url: NSURL, options: NSDataWritingOptions = []) throws {
|
||||
try storage.writeToURL(url, options: options)
|
||||
}
|
||||
|
||||
public func rangeOfData(dataToFind data: NSData, options: NSDataSearchOptions, range: Range<Int>) -> Range<Int>? {
|
||||
return storage.rangeOfData(data, options: options, range: NSRange(range)).toRange()
|
||||
}
|
||||
|
||||
public func enumerateByteRangesUsingBlock(block: (UnsafePointer<Void>, Range<Int>, inout Bool) -> Void) {
|
||||
storage.enumerateByteRangesUsingBlock { bytes, range, stop in
|
||||
var stopTmp = Bool(stop.memory)
|
||||
block(bytes, range.toRange()!, &stopTmp)
|
||||
stop.memory = ObjCBool(stopTmp)
|
||||
}
|
||||
}
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> TBinary {
|
||||
var data : NSData?
|
||||
try proto.readBinary(&data)
|
||||
return TBinary(data: data!)
|
||||
}
|
||||
|
||||
public static func writeValue(value: TBinary, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeBinary(value.storage)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension TBinary : CustomStringConvertible, CustomDebugStringConvertible {
|
||||
|
||||
public var description : String {
|
||||
return storage.description
|
||||
}
|
||||
|
||||
public var debugDescription : String {
|
||||
return storage.debugDescription
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public func ==(lhs: TBinary, rhs: TBinary) -> Bool {
|
||||
return lhs.storage == rhs.storage
|
||||
}
|
31
lib/cocoa/src/TEnum.swift
Normal file
31
lib/cocoa/src/TEnum.swift
Normal file
@ -0,0 +1,31 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public protocol TEnum : TSerializable {
|
||||
|
||||
}
|
||||
|
||||
public extension TEnum {
|
||||
|
||||
public static var thriftType : TType { return TType.I32 }
|
||||
|
||||
}
|
@ -17,7 +17,7 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import "TProtocolException.h"
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
@implementation TProtocolException
|
||||
@end
|
||||
|
||||
extern NSString *TErrorDomain;
|
@ -17,9 +17,7 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import "TException.h"
|
||||
#import "TError.h"
|
||||
|
||||
@interface TProtocolException : TException {
|
||||
}
|
||||
|
||||
@end
|
||||
NSString *TErrorDomain = @"TErrorDomain";
|
@ -1,65 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TException.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
@implementation TException
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name
|
||||
{
|
||||
return [self exceptionWithName: name reason: @"unknown" error: nil];
|
||||
}
|
||||
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name
|
||||
reason: (NSString *) reason
|
||||
{
|
||||
return [self exceptionWithName: name reason: reason error: nil];
|
||||
}
|
||||
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name
|
||||
reason: (NSString *) reason
|
||||
error: (NSError *) error
|
||||
{
|
||||
NSDictionary * userInfo = nil;
|
||||
if (error != nil) {
|
||||
userInfo = [NSDictionary dictionaryWithObject: error forKey: @"error"];
|
||||
}
|
||||
|
||||
return [super exceptionWithName: name
|
||||
reason: reason
|
||||
userInfo: userInfo];
|
||||
}
|
||||
|
||||
|
||||
- (NSString *) description
|
||||
{
|
||||
NSMutableString * result = [NSMutableString stringWithString: [self name]];
|
||||
[result appendFormat: @": %@", [self reason]];
|
||||
if ([self userInfo] != nil) {
|
||||
[result appendFormat: @"\n userInfo = %@", [self userInfo]];
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
@end
|
148
lib/cocoa/src/TList.swift
Normal file
148
lib/cocoa/src/TList.swift
Normal file
@ -0,0 +1,148 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
|
||||
public struct TList<Element : TSerializable> : MutableCollectionType, Hashable, ArrayLiteralConvertible, TSerializable {
|
||||
|
||||
public static var thriftType : TType { return .LIST }
|
||||
|
||||
typealias Storage = Array<Element>
|
||||
|
||||
public typealias Index = Storage.Index
|
||||
|
||||
private var storage = Storage()
|
||||
|
||||
public var startIndex : Index {
|
||||
return storage.startIndex
|
||||
}
|
||||
|
||||
public var endIndex : Index {
|
||||
return storage.endIndex
|
||||
}
|
||||
|
||||
public subscript (position: Index) -> Element {
|
||||
get {
|
||||
return storage[position]
|
||||
}
|
||||
set {
|
||||
storage[position] = newValue
|
||||
}
|
||||
}
|
||||
|
||||
public var hashValue : Int {
|
||||
let prime = 31
|
||||
var result = 1
|
||||
for element in storage {
|
||||
result = prime * result + element.hashValue
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
public init(arrayLiteral elements: Element...) {
|
||||
self.storage = Storage(storage)
|
||||
}
|
||||
|
||||
public init() {
|
||||
self.storage = Storage()
|
||||
}
|
||||
|
||||
public mutating func append(newElement: Element) {
|
||||
self.storage.append(newElement)
|
||||
}
|
||||
|
||||
public mutating func appendContentsOf<C : CollectionType where C.Generator.Element == Element>(newstorage: C) {
|
||||
self.storage.appendContentsOf(newstorage)
|
||||
}
|
||||
|
||||
public mutating func insert(newElement: Element, atIndex index: Int) {
|
||||
self.storage.insert(newElement, atIndex: index)
|
||||
}
|
||||
|
||||
public mutating func insertContentsOf<C : CollectionType where C.Generator.Element == Element>(newElements: C, at index: Int) {
|
||||
self.storage.insertContentsOf(newElements, at: index)
|
||||
}
|
||||
|
||||
public mutating func removeAll(keepCapacity keepCapacity: Bool = true) {
|
||||
self.storage.removeAll(keepCapacity: keepCapacity)
|
||||
}
|
||||
|
||||
public mutating func removeAtIndex(index: Index) {
|
||||
self.storage.removeAtIndex(index)
|
||||
}
|
||||
|
||||
public mutating func removeFirst(n: Int = 0) {
|
||||
self.storage.removeFirst(n)
|
||||
}
|
||||
|
||||
public mutating func removeLast() -> Element {
|
||||
return self.storage.removeLast()
|
||||
}
|
||||
|
||||
public mutating func removeRange(subRange: Range<Index>) {
|
||||
self.storage.removeRange(subRange)
|
||||
}
|
||||
|
||||
public mutating func reserveCapacity(minimumCapacity: Int) {
|
||||
self.storage.reserveCapacity(minimumCapacity)
|
||||
}
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> TList {
|
||||
let (elementType, size) = try proto.readListBegin()
|
||||
if elementType != Element.thriftType {
|
||||
throw NSError(
|
||||
domain: TProtocolErrorDomain,
|
||||
code: Int(TProtocolError.InvalidData.rawValue),
|
||||
userInfo: [TProtocolErrorExtendedErrorKey: NSNumber(int: TProtocolExtendedError.UnexpectedType.rawValue)])
|
||||
}
|
||||
var list = TList()
|
||||
for _ in 0..<size {
|
||||
let element = try Element.readValueFromProtocol(proto)
|
||||
list.storage.append(element)
|
||||
}
|
||||
try proto.readListEnd()
|
||||
return list
|
||||
}
|
||||
|
||||
public static func writeValue(value: TList, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeListBeginWithElementType(Element.thriftType, size: value.count)
|
||||
for element in value.storage {
|
||||
try Element.writeValue(element, toProtocol: proto)
|
||||
}
|
||||
try proto.writeListEnd()
|
||||
}
|
||||
}
|
||||
|
||||
extension TList : CustomStringConvertible, CustomDebugStringConvertible {
|
||||
|
||||
public var description : String {
|
||||
return storage.description
|
||||
}
|
||||
|
||||
public var debugDescription : String {
|
||||
return storage.debugDescription
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public func ==<Element>(lhs: TList<Element>, rhs: TList<Element>) -> Bool {
|
||||
return lhs.storage == rhs.storage
|
||||
}
|
158
lib/cocoa/src/TMap.swift
Normal file
158
lib/cocoa/src/TMap.swift
Normal file
@ -0,0 +1,158 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public struct TMap<Key : TSerializable, Value : TSerializable> : CollectionType, DictionaryLiteralConvertible, TSerializable {
|
||||
|
||||
public static var thriftType : TType { return .MAP }
|
||||
|
||||
typealias Storage = Dictionary<Key, Value>
|
||||
|
||||
public typealias Index = Storage.Index
|
||||
|
||||
public typealias Element = Storage.Element
|
||||
|
||||
private var storage : Storage
|
||||
|
||||
public var startIndex : Index {
|
||||
return storage.startIndex
|
||||
}
|
||||
|
||||
public var endIndex: Index {
|
||||
return storage.endIndex
|
||||
}
|
||||
|
||||
public var keys: LazyMapCollection<[Key : Value], Key> {
|
||||
return storage.keys
|
||||
}
|
||||
|
||||
public var values: LazyMapCollection<[Key : Value], Value> {
|
||||
return storage.values
|
||||
}
|
||||
|
||||
public init() {
|
||||
storage = Storage()
|
||||
}
|
||||
|
||||
public init(dictionaryLiteral elements: (Key, Value)...) {
|
||||
storage = Storage()
|
||||
for (key, value) in elements {
|
||||
storage[key] = value
|
||||
}
|
||||
}
|
||||
|
||||
public init(minimumCapacity: Int) {
|
||||
storage = Storage(minimumCapacity: minimumCapacity)
|
||||
}
|
||||
|
||||
public subscript (position: Index) -> Element {
|
||||
get {
|
||||
return storage[position]
|
||||
}
|
||||
}
|
||||
|
||||
public func indexForKey(key: Key) -> Index? {
|
||||
return storage.indexForKey(key)
|
||||
}
|
||||
|
||||
public subscript (key: Key) -> Value? {
|
||||
get {
|
||||
return storage[key]
|
||||
}
|
||||
set {
|
||||
storage[key] = newValue
|
||||
}
|
||||
}
|
||||
|
||||
public mutating func updateValue(value: Value, forKey key: Key) -> Value? {
|
||||
return updateValue(value, forKey: key)
|
||||
}
|
||||
|
||||
public mutating func removeAtIndex(index: DictionaryIndex<Key, Value>) -> (Key, Value) {
|
||||
return removeAtIndex(index)
|
||||
}
|
||||
|
||||
public mutating func removeValueForKey(key: Key) -> Value? {
|
||||
return storage.removeValueForKey(key)
|
||||
}
|
||||
|
||||
public mutating func removeAll(keepCapacity keepCapacity: Bool = false) {
|
||||
storage.removeAll(keepCapacity: keepCapacity)
|
||||
}
|
||||
|
||||
public var hashValue : Int {
|
||||
let prime = 31
|
||||
var result = 1
|
||||
for (key, value) in storage {
|
||||
result = prime * result + key.hashValue
|
||||
result = prime * result + value.hashValue
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> TMap {
|
||||
let (keyType, valueType, size) = try proto.readMapBegin()
|
||||
if keyType != Key.thriftType || valueType != Value.thriftType {
|
||||
throw NSError(
|
||||
domain: TProtocolErrorDomain,
|
||||
code: Int(TProtocolError.InvalidData.rawValue),
|
||||
userInfo: [TProtocolErrorExtendedErrorKey: NSNumber(int: TProtocolExtendedError.UnexpectedType.rawValue)])
|
||||
}
|
||||
var map = TMap()
|
||||
for _ in 0..<size {
|
||||
let key = try Key.readValueFromProtocol(proto)
|
||||
let value = try Value.readValueFromProtocol(proto)
|
||||
map.storage[key] = value
|
||||
}
|
||||
try proto.readMapEnd()
|
||||
return map
|
||||
}
|
||||
|
||||
public static func writeValue(value: TMap, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeMapBeginWithKeyType(Key.thriftType, valueType: Value.thriftType, size: value.count)
|
||||
for (key, value) in value.storage {
|
||||
try Key.writeValue(key, toProtocol: proto)
|
||||
try Value.writeValue(value, toProtocol: proto)
|
||||
}
|
||||
try proto.writeMapEnd()
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
extension TMap : CustomStringConvertible, CustomDebugStringConvertible {
|
||||
|
||||
public var description : String {
|
||||
return storage.description
|
||||
}
|
||||
|
||||
public var debugDescription : String {
|
||||
return storage.debugDescription
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public func ==<Key, Value>(lhs: TMap<Key,Value>, rhs: TMap<Key, Value>) -> Bool {
|
||||
if lhs.count != rhs.count {
|
||||
return false
|
||||
}
|
||||
return lhs.storage == rhs.storage
|
||||
}
|
@ -1,72 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
/*
|
||||
* TObjective-C.h is for supporting coexistence of both the ARC (Automatic
|
||||
* Reference Counting) mode and the Non-ARC mode of Objective-C
|
||||
* in the same source code.
|
||||
*
|
||||
* 2011/11/14 HIRANO Satoshi (AIST, Japan)
|
||||
*
|
||||
* Before:
|
||||
*
|
||||
* var = [aObject retain];
|
||||
* [aObject release];
|
||||
* [aObject autorelease];
|
||||
* [super dealloc];
|
||||
* CFFunction(obj);
|
||||
*
|
||||
* ARC and Non-ARC compatible:
|
||||
*
|
||||
* #import "TObjective-C.h"
|
||||
* var = [aObject retain_stub];
|
||||
* [aObject release_stub];
|
||||
* [aObject autorelease_stub];
|
||||
* [super dealloc_stub];
|
||||
* CFFunction(bridge_stub obj);
|
||||
*
|
||||
* Don't use retain_stub for @property(retain).
|
||||
* Use NSAutoreleasePool like this:
|
||||
* #if __has_feature(objc_arc)
|
||||
* @autoreleasepool {
|
||||
* // code
|
||||
* }
|
||||
* #else
|
||||
* NSAutoReleasePool *pool = [[NSAutoReleasePool alloc] init...
|
||||
* // code
|
||||
* [pool release];
|
||||
* #endif
|
||||
*/
|
||||
|
||||
|
||||
#if !defined(retain_stub)
|
||||
#if __has_feature(objc_arc)
|
||||
#define retain_stub self
|
||||
#define autorelease_stub self
|
||||
#define release_stub self
|
||||
#define dealloc_stub self
|
||||
#define bridge_stub __bridge
|
||||
#else
|
||||
#define retain_stub retain
|
||||
#define autorelease_stub autorelease
|
||||
#define release_stub release
|
||||
#define dealloc_stub dealloc
|
||||
#define bridge_stub
|
||||
#endif
|
||||
#endif
|
@ -20,10 +20,16 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TProtocol.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@protocol TProcessor <NSObject>
|
||||
|
||||
- (BOOL) processOnInputProtocol: (id <TProtocol>) inProtocol
|
||||
outputProtocol: (id <TProtocol>) outProtocol;
|
||||
-(BOOL) processOnInputProtocol:(id <TProtocol>)inProtocol
|
||||
outputProtocol:(id <TProtocol>)outProtocol
|
||||
error:(NSError **)error;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -20,8 +20,14 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TProcessor.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@protocol TProcessorFactory <NSObject>
|
||||
|
||||
- (id<TProcessor>) processorForTransport: (id<TTransport>) transport;
|
||||
-(id<TProcessor>) processorForTransport:(id<TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
190
lib/cocoa/src/TProtocol.swift
Normal file
190
lib/cocoa/src/TProtocol.swift
Normal file
@ -0,0 +1,190 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public extension TProtocol {
|
||||
|
||||
public func readMessageBegin() throws -> (String, TMessageType, Int) {
|
||||
|
||||
var name : NSString?
|
||||
var type : Int32 = -1
|
||||
var sequenceID : Int32 = -1
|
||||
|
||||
try readMessageBeginReturningName(&name, type: &type, sequenceID: &sequenceID)
|
||||
|
||||
return (name as String!, TMessageType(rawValue: type)!, Int(sequenceID))
|
||||
}
|
||||
|
||||
public func writeMessageBeginWithName(name: String, type: TMessageType, sequenceID: Int) throws {
|
||||
try writeMessageBeginWithName(name, type: type.rawValue, sequenceID: Int32(sequenceID))
|
||||
}
|
||||
|
||||
public func readStructBegin() throws -> (String?) {
|
||||
|
||||
var name : NSString? = nil
|
||||
|
||||
try readStructBeginReturningName(&name)
|
||||
|
||||
return (name as String?)
|
||||
}
|
||||
|
||||
public func readFieldBegin() throws -> (String?, TType, Int) {
|
||||
|
||||
var name : NSString? = nil
|
||||
var type : Int32 = -1
|
||||
var fieldID : Int32 = -1
|
||||
|
||||
try readFieldBeginReturningName(&name, type: &type, fieldID: &fieldID)
|
||||
|
||||
return (name as String?, TType(rawValue: type)!, Int(fieldID))
|
||||
}
|
||||
|
||||
public func writeFieldBeginWithName(name: String, type: TType, fieldID: Int) throws {
|
||||
try writeFieldBeginWithName(name, type: type.rawValue, fieldID: Int32(fieldID))
|
||||
}
|
||||
|
||||
public func readMapBegin() throws -> (TType, TType, Int32) {
|
||||
|
||||
var keyType : Int32 = -1
|
||||
var valueType : Int32 = -1
|
||||
var size : Int32 = 0
|
||||
|
||||
try readMapBeginReturningKeyType(&keyType, valueType: &valueType, size: &size)
|
||||
|
||||
return (TType(rawValue: keyType)!, TType(rawValue: valueType)!, size)
|
||||
}
|
||||
|
||||
public func writeMapBeginWithKeyType(keyType: TType, valueType: TType, size: Int) throws {
|
||||
try writeMapBeginWithKeyType(keyType.rawValue, valueType: valueType.rawValue, size: Int32(size))
|
||||
}
|
||||
|
||||
public func readSetBegin() throws -> (TType, Int32) {
|
||||
|
||||
var elementType : Int32 = -1
|
||||
var size : Int32 = 0
|
||||
|
||||
try readSetBeginReturningElementType(&elementType, size: &size)
|
||||
|
||||
return (TType(rawValue: elementType)!, size)
|
||||
}
|
||||
|
||||
public func writeSetBeginWithElementType(elementType: TType, size: Int) throws {
|
||||
try writeSetBeginWithElementType(elementType.rawValue, size: Int32(size))
|
||||
}
|
||||
|
||||
public func readListBegin() throws -> (TType, Int32) {
|
||||
|
||||
var elementType : Int32 = -1
|
||||
var size : Int32 = 0
|
||||
|
||||
try readListBeginReturningElementType(&elementType, size: &size)
|
||||
|
||||
return (TType(rawValue: elementType)!, size)
|
||||
}
|
||||
|
||||
public func writeListBeginWithElementType(elementType: TType, size: Int) throws {
|
||||
try writeListBeginWithElementType(elementType.rawValue, size: Int32(size))
|
||||
}
|
||||
|
||||
public func writeFieldValue<T: TSerializable>(value: T, name: String, type: TType, id: Int32) throws {
|
||||
try writeFieldBeginWithName(name, type: type.rawValue, fieldID: id)
|
||||
try writeValue(value)
|
||||
try writeFieldEnd()
|
||||
}
|
||||
|
||||
public func readValue<T: TSerializable>() throws -> T {
|
||||
return try T.readValueFromProtocol(self)
|
||||
}
|
||||
|
||||
public func writeValue<T: TSerializable>(value: T) throws {
|
||||
try T.writeValue(value, toProtocol: self)
|
||||
}
|
||||
|
||||
public func readResultMessageBegin() throws {
|
||||
|
||||
let (_, type, _) = try readMessageBegin();
|
||||
|
||||
if type == .EXCEPTION {
|
||||
let x = try readException()
|
||||
throw x
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
public func validateValue(value: Any?, named name: String) throws {
|
||||
|
||||
if value == nil {
|
||||
throw NSError(
|
||||
domain: TProtocolErrorDomain,
|
||||
code: Int(TProtocolError.Unknown.rawValue),
|
||||
userInfo: [TProtocolErrorFieldNameKey: name])
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public func readException() throws -> ErrorType {
|
||||
|
||||
var reason : String?
|
||||
var type = TApplicationError.Unknown
|
||||
|
||||
try readStructBegin()
|
||||
|
||||
fields: while (true) {
|
||||
|
||||
let (_, fieldType, fieldID) = try readFieldBegin()
|
||||
|
||||
switch (fieldID, fieldType) {
|
||||
case (_, .STOP):
|
||||
break fields
|
||||
|
||||
case (1, .STRING):
|
||||
reason = try readValue() as String
|
||||
|
||||
case (2, .I32):
|
||||
let typeVal = try readValue() as Int32
|
||||
if let tmp = TApplicationError(rawValue: typeVal) {
|
||||
type = tmp
|
||||
}
|
||||
|
||||
case let (_, unknownType):
|
||||
try skipType(unknownType)
|
||||
}
|
||||
|
||||
try readFieldEnd()
|
||||
}
|
||||
|
||||
try readStructEnd()
|
||||
|
||||
return NSError(type:type, reason:reason ?? "")
|
||||
}
|
||||
|
||||
public func writeExceptionForMessageName(name: String, sequenceID: Int, ex: NSError) throws {
|
||||
try writeMessageBeginWithName(name, type: .EXCEPTION, sequenceID: sequenceID)
|
||||
try ex.write(self)
|
||||
try writeMessageEnd()
|
||||
}
|
||||
|
||||
public func skipType(type: TType) throws {
|
||||
try TProtocolUtil.skipType(type.rawValue, onProtocol: self)
|
||||
}
|
||||
|
||||
}
|
178
lib/cocoa/src/TSerializable.swift
Normal file
178
lib/cocoa/src/TSerializable.swift
Normal file
@ -0,0 +1,178 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public protocol TSerializable : Hashable {
|
||||
|
||||
static var thriftType : TType { get }
|
||||
|
||||
init()
|
||||
|
||||
static func readValueFromProtocol(proto: TProtocol) throws -> Self
|
||||
|
||||
static func writeValue(value: Self, toProtocol proto: TProtocol) throws
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
infix operator ?== {}
|
||||
|
||||
public func ?==<T: TSerializable>(lhs: T?, rhs: T?) -> Bool {
|
||||
if let l = lhs, r = rhs {
|
||||
return l == r
|
||||
}
|
||||
return lhs == rhs
|
||||
}
|
||||
|
||||
public func ?==<T: TSerializable>(lhs: T, rhs: T) -> Bool {
|
||||
return lhs == rhs
|
||||
}
|
||||
|
||||
|
||||
|
||||
extension Bool : TSerializable {
|
||||
|
||||
public static let thriftType = TType.BOOL
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Bool {
|
||||
var value : ObjCBool = false
|
||||
try proto.readBool(&value)
|
||||
return value.boolValue
|
||||
}
|
||||
|
||||
public static func writeValue(value: Bool, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeBool(value)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Int8 : TSerializable {
|
||||
|
||||
public static let thriftType = TType.BYTE
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Int8 {
|
||||
var value = UInt8()
|
||||
try proto.readByte(&value)
|
||||
return Int8(value)
|
||||
}
|
||||
|
||||
public static func writeValue(value: Int8, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeByte(UInt8(value))
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Int16 : TSerializable {
|
||||
|
||||
public static let thriftType = TType.I16
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Int16 {
|
||||
var value = Int16()
|
||||
try proto.readI16(&value)
|
||||
return value
|
||||
}
|
||||
|
||||
public static func writeValue(value: Int16, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeI16(value)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Int : TSerializable {
|
||||
|
||||
public static let thriftType = TType.I32
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Int {
|
||||
var value = Int32()
|
||||
try proto.readI32(&value)
|
||||
return Int(value)
|
||||
}
|
||||
|
||||
public static func writeValue(value: Int, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeI32(Int32(value))
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Int32 : TSerializable {
|
||||
|
||||
public static let thriftType = TType.I32
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Int32 {
|
||||
var value = Int32()
|
||||
try proto.readI32(&value)
|
||||
return value
|
||||
}
|
||||
|
||||
public static func writeValue(value: Int32, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeI32(value)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Int64 : TSerializable {
|
||||
|
||||
public static let thriftType = TType.I64
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Int64 {
|
||||
var value = Int64()
|
||||
try proto.readI64(&value)
|
||||
return value
|
||||
}
|
||||
|
||||
public static func writeValue(value: Int64, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeI64(value)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension Double : TSerializable {
|
||||
|
||||
public static let thriftType = TType.DOUBLE
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> Double {
|
||||
var value = Double()
|
||||
try proto.readDouble(&value)
|
||||
return value
|
||||
}
|
||||
|
||||
public static func writeValue(value: Double, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeDouble(value)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension String : TSerializable {
|
||||
|
||||
public static let thriftType = TType.STRING
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> String {
|
||||
var value : NSString?
|
||||
try proto.readString(&value)
|
||||
return value as! String
|
||||
}
|
||||
|
||||
public static func writeValue(value: String, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeString(value)
|
||||
}
|
||||
|
||||
}
|
161
lib/cocoa/src/TSet.swift
Normal file
161
lib/cocoa/src/TSet.swift
Normal file
@ -0,0 +1,161 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
import Foundation
|
||||
|
||||
|
||||
public struct TSet<Element : TSerializable> : CollectionType, ArrayLiteralConvertible, TSerializable {
|
||||
|
||||
public static var thriftType : TType { return .SET }
|
||||
|
||||
public typealias Index = Storage.Index
|
||||
|
||||
typealias Storage = Set<Element>
|
||||
|
||||
private var storage : Storage
|
||||
|
||||
public init() {
|
||||
storage = Storage()
|
||||
}
|
||||
|
||||
public init(arrayLiteral elements: Element...) {
|
||||
storage = Storage(elements)
|
||||
}
|
||||
|
||||
public init<S : SequenceType where S.Generator.Element == Element>(_ sequence: S) {
|
||||
storage = Storage(sequence)
|
||||
}
|
||||
|
||||
public var startIndex : Index { return storage.startIndex }
|
||||
|
||||
public var endIndex : Index { return storage.endIndex }
|
||||
|
||||
public mutating func insert(member: Element) {
|
||||
return storage.insert(member)
|
||||
}
|
||||
|
||||
public mutating func remove(element: Element) -> Element? {
|
||||
return storage.remove(element)
|
||||
}
|
||||
|
||||
public mutating func removeAll(keepCapacity keepCapacity: Bool = false) {
|
||||
return storage.removeAll(keepCapacity: keepCapacity)
|
||||
}
|
||||
|
||||
public mutating func removeAtIndex(index: SetIndex<Element>) -> Element {
|
||||
return storage.removeAtIndex(index)
|
||||
}
|
||||
|
||||
public subscript (position: SetIndex<Element>) -> Element {
|
||||
return storage[position]
|
||||
}
|
||||
|
||||
public func union(other: TSet) -> TSet {
|
||||
return TSet(storage.union(other))
|
||||
}
|
||||
|
||||
public func intersect(other: TSet) -> TSet {
|
||||
return TSet(storage.intersect(other))
|
||||
}
|
||||
|
||||
public func exclusiveOr(other: TSet) -> TSet {
|
||||
return TSet(storage.exclusiveOr(other))
|
||||
}
|
||||
|
||||
public func subtract(other: TSet) -> TSet {
|
||||
return TSet(storage.subtract(other))
|
||||
}
|
||||
|
||||
public mutating func intersectInPlace(other: TSet) {
|
||||
storage.intersectInPlace(other)
|
||||
}
|
||||
|
||||
public mutating func exclusiveOrInPlace(other: TSet) {
|
||||
storage.exclusiveOrInPlace(other)
|
||||
}
|
||||
|
||||
public mutating func subtractInPlace(other: TSet) {
|
||||
storage.subtractInPlace(other)
|
||||
}
|
||||
|
||||
public func isSubsetOf(other: TSet) -> Bool {
|
||||
return storage.isSubsetOf(other)
|
||||
}
|
||||
|
||||
public func isDisjointWith(other: TSet) -> Bool {
|
||||
return storage.isDisjointWith(other)
|
||||
}
|
||||
|
||||
public func isSupersetOf(other: TSet) -> Bool {
|
||||
return storage.isSupersetOf(other)
|
||||
}
|
||||
|
||||
public var isEmpty: Bool { return storage.isEmpty }
|
||||
|
||||
public var hashValue : Int {
|
||||
let prime = 31
|
||||
var result = 1
|
||||
for element in storage {
|
||||
result = prime * result + element.hashValue
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
public static func readValueFromProtocol(proto: TProtocol) throws -> TSet {
|
||||
let (elementType, size) = try proto.readSetBegin()
|
||||
if elementType != Element.thriftType {
|
||||
throw NSError(
|
||||
domain: TProtocolErrorDomain,
|
||||
code: Int(TProtocolError.InvalidData.rawValue),
|
||||
userInfo: [TProtocolErrorExtendedErrorKey: NSNumber(int: elementType.rawValue)])
|
||||
}
|
||||
var set = TSet()
|
||||
for _ in 0..<size {
|
||||
let element = try Element.readValueFromProtocol(proto)
|
||||
set.storage.insert(element)
|
||||
}
|
||||
try proto.readSetEnd()
|
||||
return set
|
||||
}
|
||||
|
||||
public static func writeValue(value: TSet, toProtocol proto: TProtocol) throws {
|
||||
try proto.writeSetBeginWithElementType(Element.thriftType, size: value.count)
|
||||
for element in value.storage {
|
||||
try Element.writeValue(element, toProtocol: proto)
|
||||
}
|
||||
try proto.writeSetEnd()
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension TSet : CustomStringConvertible, CustomDebugStringConvertible {
|
||||
|
||||
public var description : String {
|
||||
return storage.description
|
||||
}
|
||||
|
||||
public var debugDescription : String {
|
||||
return storage.debugDescription
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public func ==<Element>(lhs: TSet<Element>, rhs: TSet<Element>) -> Bool {
|
||||
return lhs.storage == rhs.storage
|
||||
}
|
@ -20,8 +20,9 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TProcessorFactory.h"
|
||||
|
||||
@interface TSharedProcessorFactory : NSObject <TProcessorFactory> {
|
||||
id<TProcessor> mSharedProcessor;
|
||||
}
|
||||
- (id) initWithSharedProcessor: (id<TProcessor>) sharedProcessor;
|
||||
|
||||
@interface TSharedProcessorFactory : NSObject <TProcessorFactory>
|
||||
|
||||
-(id) initWithSharedProcessor:(id<TProcessor>)sharedProcessor;
|
||||
|
||||
@end
|
||||
|
@ -19,34 +19,31 @@
|
||||
|
||||
|
||||
#import "TSharedProcessorFactory.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
|
||||
@interface TSharedProcessorFactory ()
|
||||
|
||||
@property(strong, nonatomic) id<TProcessor> sharedProcessor;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TSharedProcessorFactory
|
||||
|
||||
|
||||
- (id) initWithSharedProcessor: (id<TProcessor>) sharedProcessor
|
||||
-(id) initWithSharedProcessor:(id<TProcessor>)sharedProcessor
|
||||
{
|
||||
self = [super init];
|
||||
if (!self) {
|
||||
return nil;
|
||||
if (self) {
|
||||
_sharedProcessor = sharedProcessor;
|
||||
}
|
||||
|
||||
mSharedProcessor = [sharedProcessor retain_stub];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
|
||||
- (void) dealloc
|
||||
-(id<TProcessor>) processorForTransport:(id<TTransport>)transport
|
||||
{
|
||||
[mSharedProcessor release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
|
||||
|
||||
- (id<TProcessor>) processorForTransport: (id<TTransport>) transport
|
||||
{
|
||||
return [[mSharedProcessor retain_stub] autorelease_stub];
|
||||
return _sharedProcessor;
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -17,14 +17,15 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import "TException.h"
|
||||
import Foundation
|
||||
|
||||
@interface TTransportException : TException {
|
||||
|
||||
public protocol TStruct : TSerializable {
|
||||
}
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
error: (NSError *) error;
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason;
|
||||
|
||||
@end
|
||||
public extension TStruct {
|
||||
|
||||
public static var thriftType : TType { return TType.STRUCT }
|
||||
|
||||
}
|
@ -26,16 +26,21 @@
|
||||
/**
|
||||
* De-serialize object from the given input protocol
|
||||
*
|
||||
* @param input protocol used for reading
|
||||
* @param input protocol used for reading
|
||||
*/
|
||||
- (void) read: (id <TProtocol>) inProtocol;
|
||||
-(BOOL) read:(id <TProtocol>)inProtocol error:(NSError **)error;
|
||||
|
||||
/**
|
||||
* Serialize object to the given protocol
|
||||
*
|
||||
* @param buf output protocol used for writing
|
||||
*/
|
||||
- (void) write: (id <TProtocol>) outProtocol;
|
||||
-(BOOL) write:(id <TProtocol>)outProtocol error:(NSError **)error;
|
||||
|
||||
|
||||
/**
|
||||
* Validate required fields
|
||||
*/
|
||||
-(BOOL) validate:(NSError *__autoreleasing *)__thriftError;
|
||||
|
||||
@end
|
||||
|
||||
|
@ -21,31 +21,29 @@
|
||||
#import "TTransport.h"
|
||||
#import "TProtocolFactory.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TBinaryProtocol : NSObject <TProtocol> {
|
||||
id <TTransport> mTransport;
|
||||
BOOL mStrictRead;
|
||||
BOOL mStrictWrite;
|
||||
int32_t mMessageSizeLimit;
|
||||
}
|
||||
|
||||
- (id) initWithTransport: (id <TTransport>) transport;
|
||||
@interface TBinaryProtocol : NSObject <TProtocol>
|
||||
|
||||
- (id) initWithTransport: (id <TTransport>) transport
|
||||
strictRead: (BOOL) strictRead
|
||||
strictWrite: (BOOL) strictWrite;
|
||||
@property (assign, nonatomic) UInt32 messageSizeLimit;
|
||||
|
||||
- (int32_t) messageSizeLimit;
|
||||
- (void) setMessageSizeLimit: (int32_t) sizeLimit;
|
||||
-(id) initWithTransport:(id <TTransport>)transport;
|
||||
|
||||
-(id) initWithTransport:(id <TTransport>)transport
|
||||
strictRead:(BOOL)strictRead
|
||||
strictWrite:(BOOL)strictWrite;
|
||||
|
||||
@end;
|
||||
|
||||
|
||||
@interface TBinaryProtocolFactory : NSObject <TProtocolFactory>
|
||||
|
||||
+(TBinaryProtocolFactory *) sharedFactory;
|
||||
|
||||
-(TBinaryProtocol *) newProtocolOnTransport:(id <TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@interface TBinaryProtocolFactory : NSObject <TProtocolFactory> {
|
||||
}
|
||||
|
||||
+ (TBinaryProtocolFactory *) sharedFactory;
|
||||
|
||||
- (TBinaryProtocol *) newProtocolOnTransport: (id <TTransport>) transport;
|
||||
|
||||
@end
|
||||
NS_ASSUME_NONNULL_END
|
File diff suppressed because it is too large
Load Diff
@ -21,16 +21,22 @@
|
||||
#import "TTransport.h"
|
||||
#import "TProtocolFactory.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@interface TCompactProtocol : NSObject <TProtocol>
|
||||
|
||||
- (id) initWithTransport: (id <TTransport>) transport;
|
||||
-(id) initWithTransport:(id <TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
@interface TCompactProtocolFactory : NSObject <TProtocolFactory>
|
||||
|
||||
+ (TCompactProtocolFactory *) sharedFactory;
|
||||
+(TCompactProtocolFactory *) sharedFactory;
|
||||
|
||||
- (TCompactProtocol *) newProtocolOnTransport: (id <TTransport>) transport;
|
||||
-(TCompactProtocol *) newProtocolOnTransport:(id <TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
File diff suppressed because it is too large
Load Diff
@ -21,13 +21,18 @@
|
||||
|
||||
#import "TProtocolDecorator.h"
|
||||
|
||||
FOUNDATION_EXPORT NSString *const MULTIPLEXED_SERVICE_SEPERATOR;
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TMultiplexedProtocol : TProtocolDecorator {
|
||||
NSString * mServiceName;
|
||||
}
|
||||
|
||||
- (id) initWithProtocol: (id <TProtocol>) protocol
|
||||
serviceName: (NSString *) name;
|
||||
extern NSString *TMultiplexedProtocolSeperator;
|
||||
|
||||
|
||||
@interface TMultiplexedProtocol : TProtocolDecorator
|
||||
|
||||
-(id) initWithProtocol:(id <TProtocol>)protocol
|
||||
serviceName:(NSString *)name;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
@ -20,48 +20,47 @@
|
||||
#import "TMultiplexedProtocol.h"
|
||||
|
||||
#import "TProtocol.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
NSString *const MULTIPLEXED_SERVICE_SEPERATOR = @":";
|
||||
NSString *TMultiplexedProtocolSeperator = @":";
|
||||
|
||||
|
||||
@interface TMultiplexedProtocol ()
|
||||
|
||||
@property(strong, nonatomic) NSString *serviceName;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TMultiplexedProtocol
|
||||
|
||||
- (id) initWithProtocol: (id <TProtocol>) protocol
|
||||
serviceName: (NSString *) name
|
||||
-(id) initWithProtocol:(id <TProtocol>)protocol
|
||||
serviceName:(NSString *)name
|
||||
{
|
||||
self = [super initWithProtocol:protocol];
|
||||
|
||||
if (self) {
|
||||
mServiceName = [name retain_stub];
|
||||
}
|
||||
return self;
|
||||
self = [super initWithProtocol:protocol];
|
||||
if (self) {
|
||||
_serviceName = name;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void) writeMessageBeginWithName: (NSString *) name
|
||||
type: (int) messageType
|
||||
sequenceID: (int) sequenceID
|
||||
-(BOOL) writeMessageBeginWithName:(NSString *)name
|
||||
type:(SInt32)messageType
|
||||
sequenceID:(SInt32)sequenceID
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
switch (messageType) {
|
||||
case TMessageType_CALL:
|
||||
case TMessageType_ONEWAY:
|
||||
{
|
||||
NSMutableString * serviceFunction = [[NSMutableString alloc] initWithString:mServiceName];
|
||||
[serviceFunction appendString:MULTIPLEXED_SERVICE_SEPERATOR];
|
||||
[serviceFunction appendString:name];
|
||||
[super writeMessageBeginWithName:serviceFunction type:messageType sequenceID:sequenceID];
|
||||
[serviceFunction release_stub];
|
||||
}
|
||||
break;
|
||||
default:
|
||||
[super writeMessageBeginWithName:name type:messageType sequenceID:sequenceID];
|
||||
break;
|
||||
}
|
||||
}
|
||||
switch (messageType) {
|
||||
case TMessageTypeCALL:
|
||||
case TMessageTypeONEWAY: {
|
||||
NSMutableString *serviceFunction = [[NSMutableString alloc] initWithString:_serviceName];
|
||||
[serviceFunction appendString:TMultiplexedProtocolSeperator];
|
||||
[serviceFunction appendString:name];
|
||||
return [super writeMessageBeginWithName:serviceFunction type:messageType sequenceID:sequenceID error:error];
|
||||
}
|
||||
break;
|
||||
|
||||
- (void) dealloc
|
||||
{
|
||||
[mServiceName release_stub];
|
||||
[super dealloc_stub];
|
||||
default:
|
||||
return [super writeMessageBeginWithName:name type:messageType sequenceID:sequenceID error:error];
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -22,127 +22,143 @@
|
||||
#import "TTransport.h"
|
||||
|
||||
|
||||
enum {
|
||||
TMessageType_CALL = 1,
|
||||
TMessageType_REPLY = 2,
|
||||
TMessageType_EXCEPTION = 3,
|
||||
TMessageType_ONEWAY = 4
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
typedef NS_ENUM (int, TMessageType) {
|
||||
TMessageTypeCALL = 1,
|
||||
TMessageTypeREPLY = 2,
|
||||
TMessageTypeEXCEPTION = 3,
|
||||
TMessageTypeONEWAY = 4
|
||||
};
|
||||
|
||||
enum {
|
||||
TType_STOP = 0,
|
||||
TType_VOID = 1,
|
||||
TType_BOOL = 2,
|
||||
TType_BYTE = 3,
|
||||
TType_DOUBLE = 4,
|
||||
TType_I16 = 6,
|
||||
TType_I32 = 8,
|
||||
TType_I64 = 10,
|
||||
TType_STRING = 11,
|
||||
TType_STRUCT = 12,
|
||||
TType_MAP = 13,
|
||||
TType_SET = 14,
|
||||
TType_LIST = 15
|
||||
typedef NS_ENUM (int, TType) {
|
||||
TTypeSTOP = 0,
|
||||
TTypeVOID = 1,
|
||||
TTypeBOOL = 2,
|
||||
TTypeBYTE = 3,
|
||||
TTypeDOUBLE = 4,
|
||||
TTypeI16 = 6,
|
||||
TTypeI32 = 8,
|
||||
TTypeI64 = 10,
|
||||
TTypeSTRING = 11,
|
||||
TTypeSTRUCT = 12,
|
||||
TTypeMAP = 13,
|
||||
TTypeSET = 14,
|
||||
TTypeLIST = 15
|
||||
};
|
||||
|
||||
|
||||
@protocol TProtocol <NSObject>
|
||||
|
||||
- (id <TTransport>) transport;
|
||||
-(id <TTransport>) transport;
|
||||
|
||||
- (void) readMessageBeginReturningName: (NSString **) name
|
||||
type: (int *) type
|
||||
sequenceID: (int *) sequenceID;
|
||||
- (void) readMessageEnd;
|
||||
-(BOOL) readMessageBeginReturningName:(NSString *__nullable __autoreleasing *__nullable)name
|
||||
type:(nullable SInt32 *)type
|
||||
sequenceID:(nullable SInt32 *)sequenceID
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readMessageEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) readStructBeginReturningName: (NSString **) name;
|
||||
- (void) readStructEnd;
|
||||
-(BOOL) readStructBeginReturningName:(NSString *__nullable __autoreleasing *__nullable)name
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readStructEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) readFieldBeginReturningName: (NSString **) name
|
||||
type: (int *) fieldType
|
||||
fieldID: (int *) fieldID;
|
||||
- (void) readFieldEnd;
|
||||
-(BOOL) readFieldBeginReturningName:(NSString *__nullable __autoreleasing *__nullable)name
|
||||
type:(SInt32 *)fieldType
|
||||
fieldID:(nullable SInt32 *)fieldID
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readFieldEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (NSString *) readString;
|
||||
-(BOOL) readString:(NSString *__nonnull __autoreleasing *__nonnull)value error:(NSError **)error;
|
||||
|
||||
- (BOOL) readBool;
|
||||
-(BOOL) readBool:(BOOL *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (unsigned char) readByte;
|
||||
-(BOOL) readByte:(UInt8 *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (short) readI16;
|
||||
-(BOOL) readI16:(SInt16 *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (int32_t) readI32;
|
||||
-(BOOL) readI32:(SInt32 *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (int64_t) readI64;
|
||||
-(BOOL) readI64:(SInt64 *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (double) readDouble;
|
||||
-(BOOL) readDouble:(double *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (NSData *) readBinary;
|
||||
-(BOOL) readBinary:(NSData *__nonnull __autoreleasing *__nonnull)value error:(NSError **)error;
|
||||
|
||||
- (void) readMapBeginReturningKeyType: (int *) keyType
|
||||
valueType: (int *) valueType
|
||||
size: (int *) size;
|
||||
- (void) readMapEnd;
|
||||
-(BOOL) readMapBeginReturningKeyType:(nullable SInt32 *)keyType
|
||||
valueType:(nullable SInt32 *)valueType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readMapEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
- (void) readSetBeginReturningElementType: (int *) elementType
|
||||
size: (int *) size;
|
||||
- (void) readSetEnd;
|
||||
-(BOOL) readSetBeginReturningElementType:(nullable SInt32 *)elementType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readSetEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
- (void) readListBeginReturningElementType: (int *) elementType
|
||||
size: (int *) size;
|
||||
- (void) readListEnd;
|
||||
-(BOOL) readListBeginReturningElementType:(nullable SInt32 *)elementType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) readListEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
- (void) writeMessageBeginWithName: (NSString *) name
|
||||
type: (int) messageType
|
||||
sequenceID: (int) sequenceID;
|
||||
- (void) writeMessageEnd;
|
||||
-(BOOL) writeMessageBeginWithName:(NSString *)name
|
||||
type:(SInt32)messageType
|
||||
sequenceID:(SInt32)sequenceID
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) writeMessageEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeStructBeginWithName: (NSString *) name;
|
||||
- (void) writeStructEnd;
|
||||
-(BOOL) writeStructBeginWithName:(NSString *)name error:(NSError **)error;
|
||||
-(BOOL) writeStructEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeFieldBeginWithName: (NSString *) name
|
||||
type: (int) fieldType
|
||||
fieldID: (int) fieldID;
|
||||
-(BOOL) writeFieldBeginWithName:(NSString *)name
|
||||
type:(SInt32)fieldType
|
||||
fieldID:(SInt32)fieldID
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeI32: (int32_t) value;
|
||||
-(BOOL) writeI32:(SInt32)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeI64: (int64_t) value;
|
||||
-(BOOL) writeI64:(SInt64)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeI16: (short) value;
|
||||
-(BOOL) writeI16:(short)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeByte: (uint8_t) value;
|
||||
-(BOOL) writeByte:(UInt8)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeString: (NSString *) value;
|
||||
-(BOOL) writeString:(NSString *)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeDouble: (double) value;
|
||||
-(BOOL) writeDouble:(double)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeBool: (BOOL) value;
|
||||
-(BOOL) writeBool:(BOOL)value error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeBinary: (NSData *) data;
|
||||
-(BOOL) writeBinary:(NSData *)data error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeFieldStop;
|
||||
-(BOOL) writeFieldStop:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeFieldEnd;
|
||||
-(BOOL) writeFieldEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeMapBeginWithKeyType: (int) keyType
|
||||
valueType: (int) valueType
|
||||
size: (int) size;
|
||||
- (void) writeMapEnd;
|
||||
-(BOOL) writeMapBeginWithKeyType:(SInt32)keyType
|
||||
valueType:(SInt32)valueType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) writeMapEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
- (void) writeSetBeginWithElementType: (int) elementType
|
||||
size: (int) size;
|
||||
- (void) writeSetEnd;
|
||||
-(BOOL) writeSetBeginWithElementType:(SInt32)elementType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
-(BOOL) writeSetEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
- (void) writeListBeginWithElementType: (int) elementType
|
||||
size: (int) size;
|
||||
-(BOOL) writeListBeginWithElementType:(SInt32)elementType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) writeListEnd;
|
||||
-(BOOL) writeListEnd:(NSError *__autoreleasing *)error;
|
||||
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -21,10 +21,14 @@
|
||||
|
||||
#import "TProtocol.h"
|
||||
|
||||
@interface TProtocolDecorator : NSObject <TProtocol> {
|
||||
id<TProtocol> mConcreteProtocol;
|
||||
}
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
- (id) initWithProtocol: (id <TProtocol>) protocol;
|
||||
|
||||
@interface TProtocolDecorator : NSObject <TProtocol>
|
||||
|
||||
-(id) initWithProtocol:(id <TProtocol>)protocol;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
@ -18,257 +18,278 @@
|
||||
*/
|
||||
|
||||
#import "TProtocolDecorator.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
|
||||
@interface TProtocolDecorator ()
|
||||
|
||||
@property(strong, nonatomic) id<TProtocol> concreteProtocol;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TProtocolDecorator
|
||||
|
||||
- (id) initWithProtocol: (id <TProtocol>) protocol
|
||||
-(id) initWithProtocol:(id <TProtocol>)protocol
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
mConcreteProtocol = [protocol retain_stub];
|
||||
}
|
||||
return self;
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_concreteProtocol = protocol;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (id <TTransport>) transport
|
||||
-(id <TTransport>) transport
|
||||
{
|
||||
return [mConcreteProtocol transport];
|
||||
return [_concreteProtocol transport];
|
||||
}
|
||||
|
||||
- (void) readMessageBeginReturningName: (NSString **) name
|
||||
type: (int *) type
|
||||
sequenceID: (int *) sequenceID
|
||||
-(BOOL) readMessageBeginReturningName:(NSString **)name
|
||||
type:(SInt32 *)type
|
||||
sequenceID:(SInt32 *)sequenceID
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readMessageBeginReturningName:name
|
||||
type:type
|
||||
sequenceID:sequenceID];
|
||||
return [_concreteProtocol readMessageBeginReturningName:name
|
||||
type:type
|
||||
sequenceID:sequenceID
|
||||
error:error];
|
||||
}
|
||||
|
||||
- (void) readMessageEnd
|
||||
-(BOOL) readMessageEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readMessageEnd];
|
||||
return [_concreteProtocol readMessageEnd:error];
|
||||
}
|
||||
|
||||
- (void) readStructBeginReturningName: (NSString **) name
|
||||
-(BOOL) readStructBeginReturningName:(NSString **)name
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readStructBeginReturningName:name];
|
||||
return [_concreteProtocol readStructBeginReturningName:name error:error];
|
||||
}
|
||||
|
||||
- (void) readStructEnd
|
||||
-(BOOL) readStructEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readStructEnd];
|
||||
return [_concreteProtocol readStructEnd:error];
|
||||
}
|
||||
|
||||
- (void) readFieldBeginReturningName: (NSString **) name
|
||||
type: (int *) fieldType
|
||||
fieldID: (int *) fieldID
|
||||
-(BOOL) readFieldBeginReturningName:(NSString **)name
|
||||
type:(SInt32 *)fieldType
|
||||
fieldID:(SInt32 *)fieldID
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readFieldBeginReturningName:name
|
||||
type:fieldType
|
||||
fieldID:fieldID];
|
||||
return [_concreteProtocol readFieldBeginReturningName:name
|
||||
type:fieldType
|
||||
fieldID:fieldID
|
||||
error:error];
|
||||
}
|
||||
- (void) readFieldEnd
|
||||
-(BOOL) readFieldEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readFieldEnd];
|
||||
return [_concreteProtocol readFieldEnd:error];
|
||||
}
|
||||
|
||||
- (NSString *) readString
|
||||
-(BOOL) readString:(NSString *__autoreleasing *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readString];
|
||||
return [_concreteProtocol readString:value error:error];
|
||||
}
|
||||
|
||||
- (BOOL) readBool
|
||||
-(BOOL) readBool:(BOOL *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readBool];
|
||||
return [_concreteProtocol readBool:value error:error];
|
||||
}
|
||||
|
||||
- (unsigned char) readByte
|
||||
-(BOOL) readByte:(UInt8 *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readByte];
|
||||
return [_concreteProtocol readByte:value error:error];
|
||||
}
|
||||
|
||||
- (short) readI16
|
||||
-(BOOL) readI16:(SInt16 *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readI16];
|
||||
return [_concreteProtocol readI16:value error:error];
|
||||
}
|
||||
|
||||
- (int32_t) readI32
|
||||
-(BOOL) readI32:(SInt32 *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readI32];
|
||||
return [_concreteProtocol readI32:value error:error];
|
||||
}
|
||||
|
||||
- (int64_t) readI64
|
||||
-(BOOL) readI64:(SInt64 *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readI64];
|
||||
return [_concreteProtocol readI64:value error:error];
|
||||
}
|
||||
|
||||
- (double) readDouble
|
||||
-(BOOL) readDouble:(double *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readDouble];
|
||||
return [_concreteProtocol readDouble:value error:error];
|
||||
}
|
||||
|
||||
- (NSData *) readBinary
|
||||
-(BOOL) readBinary:(NSData *__autoreleasing *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return [mConcreteProtocol readBinary];
|
||||
return [_concreteProtocol readBinary:value error:error];
|
||||
}
|
||||
|
||||
- (void) readMapBeginReturningKeyType: (int *) keyType
|
||||
valueType: (int *) valueType
|
||||
size: (int *) size
|
||||
-(BOOL) readMapBeginReturningKeyType:(SInt32 *)keyType
|
||||
valueType:(SInt32 *)valueType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readMapBeginReturningKeyType:keyType
|
||||
valueType:valueType
|
||||
size:size];
|
||||
return [_concreteProtocol readMapBeginReturningKeyType:keyType
|
||||
valueType:valueType
|
||||
size:size
|
||||
error:error];
|
||||
}
|
||||
- (void) readMapEnd
|
||||
-(BOOL) readMapEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readMapEnd];
|
||||
return [_concreteProtocol readMapEnd:error];
|
||||
}
|
||||
|
||||
|
||||
- (void) readSetBeginReturningElementType: (int *) elementType
|
||||
size: (int *) size
|
||||
-(BOOL) readSetBeginReturningElementType:(SInt32 *)elementType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readSetBeginReturningElementType:elementType
|
||||
size:size];
|
||||
return [_concreteProtocol readSetBeginReturningElementType:elementType
|
||||
size:size
|
||||
error:error];
|
||||
}
|
||||
- (void) readSetEnd
|
||||
-(BOOL) readSetEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readSetEnd];
|
||||
return [_concreteProtocol readSetEnd:error];
|
||||
}
|
||||
|
||||
- (void) readListBeginReturningElementType: (int *) elementType
|
||||
size: (int *) size
|
||||
-(BOOL) readListBeginReturningElementType:(SInt32 *)elementType
|
||||
size:(SInt32 *)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readListBeginReturningElementType:elementType
|
||||
size:size];
|
||||
return [_concreteProtocol readListBeginReturningElementType:elementType
|
||||
size:size
|
||||
error:error];
|
||||
}
|
||||
- (void) readListEnd
|
||||
-(BOOL) readListEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol readListEnd];
|
||||
return [_concreteProtocol readListEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeMessageBeginWithName: (NSString *) name
|
||||
type: (int) messageType
|
||||
sequenceID: (int) sequenceID
|
||||
-(BOOL) writeMessageBeginWithName:(NSString *)name
|
||||
type:(SInt32)messageType
|
||||
sequenceID:(SInt32)sequenceID
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeMessageBeginWithName:name
|
||||
type:messageType
|
||||
sequenceID:sequenceID];
|
||||
return [_concreteProtocol writeMessageBeginWithName:name
|
||||
type:messageType
|
||||
sequenceID:sequenceID
|
||||
error:error];
|
||||
}
|
||||
- (void) writeMessageEnd
|
||||
-(BOOL) writeMessageEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeMessageEnd];
|
||||
return [_concreteProtocol writeMessageEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeStructBeginWithName: (NSString *) name
|
||||
-(BOOL) writeStructBeginWithName:(NSString *)name error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeStructBeginWithName:name];
|
||||
return [_concreteProtocol writeStructBeginWithName:name error:error];
|
||||
}
|
||||
- (void) writeStructEnd
|
||||
-(BOOL) writeStructEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeStructEnd];
|
||||
return [_concreteProtocol writeStructEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeFieldBeginWithName: (NSString *) name
|
||||
type: (int) fieldType
|
||||
fieldID: (int) fieldID
|
||||
-(BOOL) writeFieldBeginWithName:(NSString *)name
|
||||
type:(SInt32)fieldType
|
||||
fieldID:(SInt32)fieldID
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeFieldBeginWithName:name
|
||||
type:fieldType
|
||||
fieldID:fieldID];
|
||||
return [_concreteProtocol writeFieldBeginWithName:name
|
||||
type:fieldType
|
||||
fieldID:fieldID
|
||||
error:error];
|
||||
}
|
||||
|
||||
- (void) writeI32: (int32_t) value
|
||||
-(BOOL) writeI32:(SInt32)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeI32:value];
|
||||
return [_concreteProtocol writeI32:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeI64: (int64_t) value
|
||||
-(BOOL) writeI64:(SInt64)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeI64:value];
|
||||
return [_concreteProtocol writeI64:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeI16: (short) value
|
||||
-(BOOL) writeI16:(SInt16)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeI16:value];
|
||||
return [_concreteProtocol writeI16:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeByte: (uint8_t) value
|
||||
-(BOOL) writeByte:(UInt8)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeByte:value];
|
||||
return [_concreteProtocol writeByte:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeString: (NSString *) value
|
||||
-(BOOL) writeString:(NSString *)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeString:value];
|
||||
return [_concreteProtocol writeString:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeDouble: (double) value
|
||||
-(BOOL) writeDouble:(double)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeDouble:value];
|
||||
return [_concreteProtocol writeDouble:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeBool: (BOOL) value
|
||||
-(BOOL) writeBool:(BOOL)value error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeBool:value];
|
||||
return [_concreteProtocol writeBool:value error:error];
|
||||
}
|
||||
|
||||
- (void) writeBinary: (NSData *) data
|
||||
-(BOOL) writeBinary:(NSData *)data error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeBinary:data];
|
||||
return [_concreteProtocol writeBinary:data error:error];
|
||||
}
|
||||
|
||||
- (void) writeFieldStop
|
||||
-(BOOL) writeFieldStop:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeFieldStop];
|
||||
return [_concreteProtocol writeFieldStop:error];
|
||||
}
|
||||
|
||||
- (void) writeFieldEnd
|
||||
-(BOOL) writeFieldEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeFieldEnd];
|
||||
return [_concreteProtocol writeFieldEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeMapBeginWithKeyType: (int) keyType
|
||||
valueType: (int) valueType
|
||||
size: (int) size
|
||||
-(BOOL) writeMapBeginWithKeyType:(SInt32)keyType
|
||||
valueType:(SInt32)valueType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeMapBeginWithKeyType:keyType
|
||||
valueType:valueType
|
||||
size:size];
|
||||
}
|
||||
- (void) writeMapEnd
|
||||
{
|
||||
[mConcreteProtocol writeMapEnd];
|
||||
return [_concreteProtocol writeMapBeginWithKeyType:keyType
|
||||
valueType:valueType
|
||||
size:size
|
||||
error:error];
|
||||
}
|
||||
|
||||
- (void) writeSetBeginWithElementType: (int) elementType
|
||||
size: (int) size
|
||||
-(BOOL) writeMapEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeSetBeginWithElementType:elementType size:size];
|
||||
return [_concreteProtocol writeMapEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeSetEnd
|
||||
-(BOOL) writeSetBeginWithElementType:(SInt32)elementType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeSetEnd];
|
||||
return [_concreteProtocol writeSetBeginWithElementType:elementType size:size error:error];
|
||||
}
|
||||
|
||||
- (void) writeListBeginWithElementType: (int) elementType
|
||||
size: (int) size
|
||||
-(BOOL) writeSetEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeListBeginWithElementType:elementType size:size];
|
||||
return [_concreteProtocol writeSetEnd:error];
|
||||
}
|
||||
|
||||
- (void) writeListEnd
|
||||
-(BOOL) writeListBeginWithElementType:(SInt32)elementType
|
||||
size:(SInt32)size
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol writeListEnd];
|
||||
return [_concreteProtocol writeListBeginWithElementType:elementType size:size error:error];
|
||||
}
|
||||
|
||||
- (void) dealloc
|
||||
-(BOOL) writeListEnd:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mConcreteProtocol release_stub];
|
||||
[super dealloc_stub];
|
||||
return [_concreteProtocol writeListEnd:error];
|
||||
}
|
||||
|
||||
@end
|
||||
|
76
lib/cocoa/src/protocol/TProtocolError.h
Normal file
76
lib/cocoa/src/protocol/TProtocolError.h
Normal file
@ -0,0 +1,76 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TError.h"
|
||||
|
||||
|
||||
extern NSString *TProtocolErrorDomain;
|
||||
|
||||
typedef NS_ENUM (int, TProtocolError) {
|
||||
TProtocolErrorUnknown = 0,
|
||||
TProtocolErrorInvalidData = 1,
|
||||
TProtocolErrorNegativeSize = 2,
|
||||
TProtocolErrorSizeLimit = 3,
|
||||
TProtocolErrorBadVersion = 4,
|
||||
TProtocolErrorNotImplemented = 5,
|
||||
TProtocolErrorDepthLimit = 6,
|
||||
};
|
||||
|
||||
|
||||
typedef NS_ENUM(int, TProtocolExtendedError) {
|
||||
TProtocolExtendedErrorMissingRequiredField = 1001,
|
||||
TProtocolExtendedErrorUnexpectedType = 1002,
|
||||
TProtocolExtendedErrorMismatchedProtocol = 1003,
|
||||
};
|
||||
|
||||
extern NSString *TProtocolErrorExtendedErrorKey;
|
||||
extern NSString *TProtocolErrorFieldNameKey;
|
||||
extern NSString *TProtocolErrorExpectedIdKey;
|
||||
extern NSString *TProtocolErrorExpectedVersionKey;
|
||||
extern NSString *TProtocolErrorTypeKey;
|
||||
extern NSString *TProtocolErrorSourceLineKey;
|
||||
extern NSString *TProtocolErrorSourceFileKey;
|
||||
extern NSString *TProtocolErrorSourceMethodKey;
|
||||
extern NSString *TProtocolErrorMessageNameKey;
|
||||
|
||||
|
||||
#define PROTOCOL_ERROR(ret, err, ...) \
|
||||
if (error) { \
|
||||
*error = [NSError errorWithDomain:TProtocolErrorDomain \
|
||||
code:TProtocolError ## err \
|
||||
userInfo:@{NSLocalizedDescriptionKey: [NSString stringWithFormat:__VA_ARGS__], \
|
||||
@"SourceFile": [NSString stringWithUTF8String:__FILE__], \
|
||||
@"SourceLine": @(__LINE__), \
|
||||
@"SourceFunction": [NSString stringWithUTF8String:__PRETTY_FUNCTION__], \
|
||||
@"Message": self.currentMessageName ? self.currentMessageName : @""}]; \
|
||||
} \
|
||||
return ret
|
||||
|
||||
#define PROTOCOL_TRANSPORT_ERROR(ret, errorPtr, ...) \
|
||||
if (errorPtr) { \
|
||||
*error = [NSError errorWithDomain:TProtocolErrorDomain \
|
||||
code:TProtocolErrorUnknown \
|
||||
userInfo:@{NSLocalizedDescriptionKey: [[NSString stringWithFormat:__VA_ARGS__] stringByAppendingFormat:@": %@", [(*errorPtr) localizedDescription]], \
|
||||
TProtocolErrorSourceFileKey: [NSString stringWithUTF8String:__FILE__], \
|
||||
TProtocolErrorSourceLineKey: @(__LINE__), \
|
||||
TProtocolErrorSourceMethodKey: [NSString stringWithUTF8String:__PRETTY_FUNCTION__], \
|
||||
TProtocolErrorMessageNameKey: self.currentMessageName ? self.currentMessageName : @"", \
|
||||
NSUnderlyingErrorKey: *errorPtr}]; \
|
||||
} \
|
||||
return ret
|
@ -17,24 +17,17 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TNSStreamTransport.h"
|
||||
#import "TProtocolError.h"
|
||||
|
||||
@interface TSSLSocketClient : TNSStreamTransport
|
||||
#if TARGET_OS_IPHONE || (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_6)
|
||||
<NSStreamDelegate>
|
||||
#endif
|
||||
{
|
||||
NSInputStream *inputStream;
|
||||
NSOutputStream *outputStream;
|
||||
@private
|
||||
NSString *sslHostname;
|
||||
int sd;
|
||||
}
|
||||
|
||||
- (id) initWithHostname: (NSString *) hostname
|
||||
port: (int) port;
|
||||
NSString *TProtocolErrorDomain = @"TProtocolErrorDomain";
|
||||
|
||||
- (BOOL) isOpen;
|
||||
|
||||
@end
|
||||
NSString *TProtocolErrorExtendedErrorKey = @"extendedError";
|
||||
NSString *TProtocolErrorFieldNameKey = @"field";
|
||||
NSString *TProtocolErrorExpectedIdKey = @"expectedId";
|
||||
NSString *TProtocolErrorExpectedVersionKey = @"expectedVersion";
|
||||
NSString *TProtocolErrorTypeKey = @"type";
|
||||
NSString *TProtocolErrorSourceLineKey = @"sourceLine";
|
||||
NSString *TProtocolErrorSourceFileKey = @"sourceFile";
|
||||
NSString *TProtocolErrorSourceMethodKey = @"sourceMethod";
|
||||
NSString *TProtocolErrorMessageNameKey = @"messageName";
|
@ -21,9 +21,16 @@
|
||||
#import "TProtocol.h"
|
||||
#import "TTransport.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@protocol TProtocolFactory <NSObject>
|
||||
|
||||
- (id <TProtocol>) newProtocolOnTransport: (id <TTransport>) transport;
|
||||
@property (readonly, nonatomic) NSString *protocolName;
|
||||
|
||||
-(id<TProtocol>) newProtocolOnTransport:(id<TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -20,10 +20,14 @@
|
||||
#import "TProtocol.h"
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TProtocolUtil : NSObject {
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
}
|
||||
|
||||
+ (void) skipType: (int) type onProtocol: (id <TProtocol>) protocol;
|
||||
@interface TProtocolUtil : NSObject
|
||||
|
||||
@end
|
||||
+(BOOL) skipType:(int)type onProtocol:(id <TProtocol>)protocol error:(NSError **)error;
|
||||
|
||||
@end;
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
@ -21,84 +21,151 @@
|
||||
|
||||
@implementation TProtocolUtil
|
||||
|
||||
+ (void) skipType: (int) type onProtocol: (id <TProtocol>) protocol
|
||||
+(BOOL) skipType:(int)type onProtocol:(id <TProtocol>)protocol error:(NSError **)error
|
||||
{
|
||||
switch (type) {
|
||||
case TType_BOOL:
|
||||
[protocol readBool];
|
||||
break;
|
||||
case TType_BYTE:
|
||||
[protocol readByte];
|
||||
break;
|
||||
case TType_I16:
|
||||
[protocol readI16];
|
||||
break;
|
||||
case TType_I32:
|
||||
[protocol readI32];
|
||||
break;
|
||||
case TType_I64:
|
||||
[protocol readI64];
|
||||
break;
|
||||
case TType_DOUBLE:
|
||||
[protocol readDouble];
|
||||
break;
|
||||
case TType_STRING:
|
||||
[protocol readString];
|
||||
break;
|
||||
case TType_STRUCT:
|
||||
[protocol readStructBeginReturningName: NULL];
|
||||
case TTypeBOOL: {
|
||||
BOOL val;
|
||||
if (![protocol readBool:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeBYTE: {
|
||||
UInt8 val;
|
||||
if (![protocol readByte:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeI16: {
|
||||
SInt16 val;
|
||||
if (![protocol readI16:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeI32: {
|
||||
SInt32 val;
|
||||
if (![protocol readI32:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeI64: {
|
||||
SInt64 val;
|
||||
if (![protocol readI64:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeDOUBLE: {
|
||||
double val;
|
||||
if (![protocol readDouble:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeSTRING: {
|
||||
NSString *val;
|
||||
if (![protocol readString:&val error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeSTRUCT: {
|
||||
if (![protocol readStructBeginReturningName:NULL error:error]) {
|
||||
return NO;
|
||||
}
|
||||
while (true) {
|
||||
int fieldType;
|
||||
[protocol readFieldBeginReturningName: nil type: &fieldType fieldID: nil];
|
||||
if (fieldType == TType_STOP) {
|
||||
SInt32 fieldType;
|
||||
if (![protocol readFieldBeginReturningName:nil type:&fieldType fieldID:nil error:error]) {
|
||||
return NO;
|
||||
}
|
||||
if (fieldType == TTypeSTOP) {
|
||||
break;
|
||||
}
|
||||
[TProtocolUtil skipType: fieldType onProtocol: protocol];
|
||||
[protocol readFieldEnd];
|
||||
if (![self skipType:fieldType onProtocol:protocol error:error]) {
|
||||
return NO;
|
||||
}
|
||||
if (![protocol readFieldEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
if (![protocol readStructEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeMAP: {
|
||||
SInt32 keyType;
|
||||
SInt32 valueType;
|
||||
SInt32 size;
|
||||
if (![protocol readMapBeginReturningKeyType:&keyType valueType:&valueType size:&size error:error]) {
|
||||
return NO;
|
||||
}
|
||||
[protocol readStructEnd];
|
||||
break;
|
||||
case TType_MAP:
|
||||
{
|
||||
int keyType;
|
||||
int valueType;
|
||||
int size;
|
||||
[protocol readMapBeginReturningKeyType: &keyType valueType: &valueType size: &size];
|
||||
int i;
|
||||
for (i = 0; i < size; i++) {
|
||||
[TProtocolUtil skipType: keyType onProtocol: protocol];
|
||||
[TProtocolUtil skipType: valueType onProtocol: protocol];
|
||||
}
|
||||
[protocol readMapEnd];
|
||||
}
|
||||
break;
|
||||
case TType_SET:
|
||||
{
|
||||
int elemType;
|
||||
int size;
|
||||
[protocol readSetBeginReturningElementType: &elemType size: &size];
|
||||
int i;
|
||||
for (i = 0; i < size; i++) {
|
||||
[TProtocolUtil skipType: elemType onProtocol: protocol];
|
||||
if (![TProtocolUtil skipType:keyType onProtocol:protocol error:error]) {
|
||||
return NO;
|
||||
}
|
||||
[protocol readSetEnd];
|
||||
}
|
||||
break;
|
||||
case TType_LIST:
|
||||
{
|
||||
int elemType;
|
||||
int size;
|
||||
[protocol readListBeginReturningElementType: &elemType size: &size];
|
||||
int i;
|
||||
for (i = 0; i < size; i++) {
|
||||
[TProtocolUtil skipType: elemType onProtocol: protocol];
|
||||
if (![TProtocolUtil skipType:valueType onProtocol:protocol error:error]) {
|
||||
return NO;
|
||||
}
|
||||
[protocol readListEnd];
|
||||
}
|
||||
break;
|
||||
default:
|
||||
return;
|
||||
if (![protocol readMapEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeSET: {
|
||||
SInt32 elemType;
|
||||
SInt32 size;
|
||||
if (![protocol readSetBeginReturningElementType:&elemType size:&size error:error]) {
|
||||
return NO;
|
||||
}
|
||||
int i;
|
||||
for (i = 0; i < size; i++) {
|
||||
if (![TProtocolUtil skipType:elemType onProtocol:protocol error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
if (![protocol readSetEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case TTypeLIST: {
|
||||
SInt32 elemType;
|
||||
SInt32 size;
|
||||
if (![protocol readListBeginReturningElementType:&elemType size:&size error:error]) {
|
||||
return NO;
|
||||
}
|
||||
int i;
|
||||
for (i = 0; i < size; i++) {
|
||||
if (![TProtocolUtil skipType:elemType onProtocol:protocol error:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
if (![protocol readListEnd:error]) {
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -27,23 +27,21 @@
|
||||
#import <CFNetwork/CFNetwork.h>
|
||||
#endif
|
||||
|
||||
extern NSString * const kTSocketServer_ClientConnectionFinishedForProcessorNotification;
|
||||
extern NSString * const kTSocketServer_ProcessorKey;
|
||||
extern NSString * const kTSockerServer_TransportKey;
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@interface TSocketServer : NSObject {
|
||||
NSFileHandle * mSocketFileHandle;
|
||||
id <TProtocolFactory> mInputProtocolFactory;
|
||||
id <TProtocolFactory> mOutputProtocolFactory;
|
||||
id <TProcessorFactory> mProcessorFactory;
|
||||
}
|
||||
extern NSString *const TSocketServerClientConnectionFinished;
|
||||
extern NSString *const TSocketServerProcessorKey;
|
||||
extern NSString *const TSockerServerTransportKey;
|
||||
|
||||
- (id) initWithPort: (int) port
|
||||
protocolFactory: (id <TProtocolFactory>) protocolFactory
|
||||
processorFactory: (id <TProcessorFactory>) processorFactory;
|
||||
|
||||
@interface TSocketServer : NSObject
|
||||
|
||||
-(instancetype) initWithPort:(int)port
|
||||
protocolFactory:(id <TProtocolFactory>)protocolFactory
|
||||
processorFactory:(id <TProcessorFactory>)processorFactory;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -21,37 +21,51 @@
|
||||
#import "TSocketServer.h"
|
||||
#import "TNSFileHandleTransport.h"
|
||||
#import "TProtocol.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
#import <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
|
||||
|
||||
|
||||
NSString * const kTSocketServer_ClientConnectionFinishedForProcessorNotification = @"TSocketServer_ClientConnectionFinishedForProcessorNotification";
|
||||
NSString * const kTSocketServer_ProcessorKey = @"TSocketServer_Processor";
|
||||
NSString * const kTSockerServer_TransportKey = @"TSockerServer_Transport";
|
||||
NSString *const TSocketServerClientConnectionFinished = @"TSocketServerClientConnectionFinished";
|
||||
NSString *const TSocketServerProcessorKey = @"TSocketServerProcessor";
|
||||
NSString *const TSockerServerTransportKey = @"TSockerServerTransport";
|
||||
|
||||
|
||||
@interface TSocketServer ()
|
||||
|
||||
@property(strong, nonatomic) id<TProtocolFactory> inputProtocolFactory;
|
||||
@property(strong, nonatomic) id<TProtocolFactory> outputProtocolFactory;
|
||||
@property(strong, nonatomic) id<TProcessorFactory> processorFactory;
|
||||
@property(strong, nonatomic) NSFileHandle *socketFileHandle;
|
||||
@property(strong, nonatomic) dispatch_queue_t processingQueue;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TSocketServer
|
||||
|
||||
- (id) initWithPort: (int) port
|
||||
protocolFactory: (id <TProtocolFactory>) protocolFactory
|
||||
processorFactory: (id <TProcessorFactory>) processorFactory
|
||||
-(instancetype) initWithPort:(int)port
|
||||
protocolFactory:(id <TProtocolFactory>)protocolFactory
|
||||
processorFactory:(id <TProcessorFactory>)processorFactory;
|
||||
{
|
||||
self = [super init];
|
||||
|
||||
mInputProtocolFactory = [protocolFactory retain_stub];
|
||||
mOutputProtocolFactory = [protocolFactory retain_stub];
|
||||
mProcessorFactory = [processorFactory retain_stub];
|
||||
_inputProtocolFactory = protocolFactory;
|
||||
_outputProtocolFactory = protocolFactory;
|
||||
_processorFactory = processorFactory;
|
||||
|
||||
dispatch_queue_attr_t processingQueueAttr =
|
||||
dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_CONCURRENT, QOS_CLASS_BACKGROUND, 0);
|
||||
|
||||
_processingQueue = dispatch_queue_create("TSocketServer.processing", processingQueueAttr);
|
||||
|
||||
// create a socket.
|
||||
int fd = -1;
|
||||
CFSocketRef socket = CFSocketCreate(kCFAllocatorDefault, PF_INET, SOCK_STREAM, IPPROTO_TCP, 0, NULL, NULL);
|
||||
if (socket) {
|
||||
CFOptionFlags flagsToClear = kCFSocketCloseOnInvalidate;
|
||||
CFSocketSetSocketFlags(socket, CFSocketGetSocketFlags(socket) & ~flagsToClear);
|
||||
|
||||
CFSocketSetSocketFlags(socket, CFSocketGetSocketFlags(socket) & ~kCFSocketCloseOnInvalidate);
|
||||
fd = CFSocketGetNative(socket);
|
||||
int yes = 1;
|
||||
setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (void *)&yes, sizeof(yes));
|
||||
@ -63,135 +77,88 @@ NSString * const kTSockerServer_TransportKey = @"TSockerServer_Transport";
|
||||
addr.sin_port = htons(port);
|
||||
addr.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||
NSData *address = [NSData dataWithBytes:&addr length:sizeof(addr)];
|
||||
if (CFSocketSetAddress(socket, (bridge_stub CFDataRef)address) != kCFSocketSuccess) {
|
||||
if (CFSocketSetAddress(socket, (__bridge CFDataRef)address) != kCFSocketSuccess) {
|
||||
CFSocketInvalidate(socket);
|
||||
CFRelease(socket);
|
||||
NSLog(@"*** Could not bind to address");
|
||||
NSLog(@"TSocketServer: Could not bind to address");
|
||||
return nil;
|
||||
}
|
||||
} else {
|
||||
NSLog(@"*** No server socket");
|
||||
}
|
||||
else {
|
||||
NSLog(@"TSocketServer: No server socket");
|
||||
return nil;
|
||||
}
|
||||
|
||||
|
||||
// wrap it in a file handle so we can get messages from it
|
||||
mSocketFileHandle = [[NSFileHandle alloc] initWithFileDescriptor: fd
|
||||
closeOnDealloc: YES];
|
||||
|
||||
_socketFileHandle = [[NSFileHandle alloc] initWithFileDescriptor:fd
|
||||
closeOnDealloc:YES];
|
||||
|
||||
// throw away our socket
|
||||
CFSocketInvalidate(socket);
|
||||
CFRelease(socket);
|
||||
|
||||
// register for notifications of accepted incoming connections
|
||||
[[NSNotificationCenter defaultCenter] addObserver: self
|
||||
selector: @selector(connectionAccepted:)
|
||||
name: NSFileHandleConnectionAcceptedNotification
|
||||
object: mSocketFileHandle];
|
||||
|
||||
|
||||
// register for notifications of accepted incoming connections
|
||||
[[NSNotificationCenter defaultCenter] addObserver:self
|
||||
selector:@selector(connectionAccepted:)
|
||||
name:NSFileHandleConnectionAcceptedNotification
|
||||
object:_socketFileHandle];
|
||||
|
||||
// tell socket to listen
|
||||
[mSocketFileHandle acceptConnectionInBackgroundAndNotify];
|
||||
|
||||
NSLog(@"Listening on TCP port %d", port);
|
||||
|
||||
[_socketFileHandle acceptConnectionInBackgroundAndNotify];
|
||||
|
||||
NSLog(@"TSocketServer: Listening on TCP port %d", port);
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
|
||||
- (void) dealloc {
|
||||
-(void) dealloc
|
||||
{
|
||||
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
||||
[mInputProtocolFactory release_stub];
|
||||
[mOutputProtocolFactory release_stub];
|
||||
[mProcessorFactory release_stub];
|
||||
[mSocketFileHandle release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
|
||||
|
||||
- (void) connectionAccepted: (NSNotification *) aNotification
|
||||
-(void) connectionAccepted:(NSNotification *)notification
|
||||
{
|
||||
NSFileHandle * socket = [[aNotification userInfo] objectForKey: NSFileHandleNotificationFileHandleItem];
|
||||
NSFileHandle *socket = [notification.userInfo objectForKey:NSFileHandleNotificationFileHandleItem];
|
||||
|
||||
// now that we have a client connected, spin off a thread to handle activity
|
||||
[NSThread detachNewThreadSelector: @selector(handleClientConnection:)
|
||||
toTarget: self
|
||||
withObject: socket];
|
||||
// Now that we have a client connected, handle request on queue
|
||||
dispatch_async(_processingQueue, ^{
|
||||
|
||||
[[aNotification object] acceptConnectionInBackgroundAndNotify];
|
||||
[self handleClientConnection:socket];
|
||||
|
||||
});
|
||||
|
||||
// Continue accepting connections
|
||||
[_socketFileHandle acceptConnectionInBackgroundAndNotify];
|
||||
}
|
||||
|
||||
|
||||
- (void) handleClientConnection: (NSFileHandle *) clientSocket
|
||||
-(void) handleClientConnection:(NSFileHandle *)clientSocket
|
||||
{
|
||||
#if __has_feature(objc_arc)
|
||||
@autoreleasepool {
|
||||
TNSFileHandleTransport * transport = [[TNSFileHandleTransport alloc] initWithFileHandle: clientSocket];
|
||||
id<TProcessor> processor = [mProcessorFactory processorForTransport: transport];
|
||||
|
||||
id <TProtocol> inProtocol = [mInputProtocolFactory newProtocolOnTransport: transport];
|
||||
id <TProtocol> outProtocol = [mOutputProtocolFactory newProtocolOnTransport: transport];
|
||||
|
||||
@try {
|
||||
BOOL result = NO;
|
||||
do {
|
||||
@autoreleasepool {
|
||||
result = [processor processOnInputProtocol: inProtocol outputProtocol: outProtocol];
|
||||
}
|
||||
} while (result);
|
||||
}
|
||||
@catch (TTransportException * te) {
|
||||
(void)te;
|
||||
//NSLog(@"Caught transport exception, abandoning client connection: %@", te);
|
||||
}
|
||||
|
||||
NSNotification * n = [NSNotification notificationWithName: kTSocketServer_ClientConnectionFinishedForProcessorNotification
|
||||
object: self
|
||||
userInfo: [NSDictionary dictionaryWithObjectsAndKeys:
|
||||
processor,
|
||||
kTSocketServer_ProcessorKey,
|
||||
transport,
|
||||
kTSockerServer_TransportKey,
|
||||
nil]];
|
||||
[[NSNotificationCenter defaultCenter] performSelectorOnMainThread: @selector(postNotification:) withObject: n waitUntilDone: YES];
|
||||
|
||||
@autoreleasepool {
|
||||
|
||||
TNSFileHandleTransport *transport = [[TNSFileHandleTransport alloc] initWithFileHandle:clientSocket];
|
||||
id<TProcessor> processor = [_processorFactory processorForTransport:transport];
|
||||
|
||||
id <TProtocol> inProtocol = [_inputProtocolFactory newProtocolOnTransport:transport];
|
||||
id <TProtocol> outProtocol = [_outputProtocolFactory newProtocolOnTransport:transport];
|
||||
|
||||
NSError *error;
|
||||
if (![processor processOnInputProtocol:inProtocol outputProtocol:outProtocol error:&error]) {
|
||||
// Handle error
|
||||
NSLog(@"Error processing request: %@", error);
|
||||
}
|
||||
#else
|
||||
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
|
||||
|
||||
TNSFileHandleTransport * transport = [[TNSFileHandleTransport alloc] initWithFileHandle: clientSocket];
|
||||
id<TProcessor> processor = [mProcessorFactory processorForTransport: transport];
|
||||
|
||||
id <TProtocol> inProtocol = [[mInputProtocolFactory newProtocolOnTransport: transport] autorelease];
|
||||
id <TProtocol> outProtocol = [[mOutputProtocolFactory newProtocolOnTransport: transport] autorelease];
|
||||
|
||||
@try {
|
||||
BOOL result = NO;
|
||||
do {
|
||||
NSAutoreleasePool * myPool = [[NSAutoreleasePool alloc] init];
|
||||
result = [processor processOnInputProtocol: inProtocol outputProtocol: outProtocol];
|
||||
[myPool release];
|
||||
} while (result);
|
||||
}
|
||||
@catch (TTransportException * te) {
|
||||
//NSLog(@"Caught transport exception, abandoning client connection: %@", te);
|
||||
}
|
||||
dispatch_async(dispatch_get_main_queue(), ^{
|
||||
|
||||
NSNotification * n = [NSNotification notificationWithName: kTSocketServer_ClientConnectionFinishedForProcessorNotification
|
||||
object: self
|
||||
userInfo: [NSDictionary dictionaryWithObjectsAndKeys:
|
||||
processor,
|
||||
kTSocketServer_ProcessorKey,
|
||||
transport,
|
||||
kTSockerServer_TransportKey,
|
||||
nil]];
|
||||
[[NSNotificationCenter defaultCenter] performSelectorOnMainThread: @selector(postNotification:) withObject: n waitUntilDone: YES];
|
||||
|
||||
[pool release];
|
||||
#endif
|
||||
[NSNotificationCenter.defaultCenter postNotificationName:TSocketServerClientConnectionFinished
|
||||
object:self
|
||||
userInfo:@{TSocketServerProcessorKey: processor,
|
||||
TSockerServerTransportKey: transport}];
|
||||
});
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
||||
|
@ -18,12 +18,29 @@
|
||||
*/
|
||||
|
||||
#import "TTransport.h"
|
||||
#import "TException.h"
|
||||
|
||||
typedef void(^TAsyncFailureBlock)(TException *);
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@protocol TAsyncTransport;
|
||||
|
||||
|
||||
@protocol TAsyncTransportFactory <NSObject>
|
||||
|
||||
-(id<TAsyncTransport>) newTransport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
typedef void (^TAsyncCompletionBlock)();
|
||||
typedef void (^TAsyncFailureBlock)(NSError * __nonnull);
|
||||
|
||||
|
||||
@protocol TAsyncTransport <TTransport>
|
||||
|
||||
- (void) flush:(dispatch_block_t)flushed failure:(TAsyncFailureBlock)failure;
|
||||
-(void) flushWithCompletion:(TAsyncCompletionBlock)completed failure:(TAsyncFailureBlock)failure;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
@ -20,10 +20,14 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TFramedTransport : NSObject <TTransport> {
|
||||
id <TTransport> mTransport;
|
||||
}
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
- (id) initWithTransport: (id <TTransport>) transport;
|
||||
|
||||
@interface TFramedTransport : NSObject <TTransport>
|
||||
|
||||
-(id) initWithTransport:(id <TTransport>)transport;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -18,126 +18,163 @@
|
||||
*/
|
||||
|
||||
#import "TFramedTransport.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
#define HEADER_SIZE 4
|
||||
#define INIT_FRAME_SIZE 1024
|
||||
|
||||
@implementation TFramedTransport {
|
||||
NSMutableData* writeBuffer;
|
||||
NSMutableData* readBuffer;
|
||||
NSUInteger readOffset;
|
||||
uint8_t dummy_header[HEADER_SIZE];
|
||||
|
||||
@interface TFramedTransport ()
|
||||
|
||||
@property(strong, nonatomic) id<TTransport> transport;
|
||||
@property(strong, nonatomic) NSMutableData *writeBuffer;
|
||||
@property(strong, nonatomic) NSMutableData *readBuffer;
|
||||
@property(assign, nonatomic) NSUInteger readOffset;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TFramedTransport
|
||||
|
||||
-(id) initWithTransport:(id <TTransport>)aTransport
|
||||
{
|
||||
if ((self = [self init])) {
|
||||
_transport = aTransport;
|
||||
_readBuffer = nil;
|
||||
_readOffset = 0;
|
||||
_writeBuffer = [NSMutableData dataWithLength:HEADER_SIZE];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (id) initWithTransport:(id <TTransport>)transport
|
||||
-(BOOL) flush:(NSError **)error
|
||||
{
|
||||
mTransport = [transport retain_stub];
|
||||
readBuffer = nil;
|
||||
readOffset = 0;
|
||||
writeBuffer = [[NSMutableData alloc] initWithCapacity:INIT_FRAME_SIZE];
|
||||
[writeBuffer appendBytes:dummy_header length:HEADER_SIZE];
|
||||
return self;
|
||||
int len = (int)[_writeBuffer length];
|
||||
int data_len = len - HEADER_SIZE;
|
||||
if (data_len < 0) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
UInt8 i32rd[HEADER_SIZE];
|
||||
i32rd[0] = (UInt8)(0xff & (data_len >> 24));
|
||||
i32rd[1] = (UInt8)(0xff & (data_len >> 16));
|
||||
i32rd[2] = (UInt8)(0xff & (data_len >> 8));
|
||||
i32rd[3] = (UInt8)(0xff & (data_len));
|
||||
|
||||
// should we make a copy of the writeBuffer instead? Better for threaded
|
||||
// operations!
|
||||
[_writeBuffer replaceBytesInRange:NSMakeRange(0, HEADER_SIZE)
|
||||
withBytes:i32rd length:HEADER_SIZE];
|
||||
|
||||
if (![_transport write:_writeBuffer.mutableBytes offset:0 length:len error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![_transport flush:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
_writeBuffer.length = HEADER_SIZE;
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void) dealloc
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[mTransport release_stub];
|
||||
[writeBuffer release_stub];
|
||||
if (readBuffer != nil)
|
||||
[readBuffer release_stub];
|
||||
[super dealloc_stub];
|
||||
[_writeBuffer appendBytes:data+offset length:length];
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void)flush
|
||||
-(BOOL) readAll:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
size_t headerAndDataLength = [writeBuffer length];
|
||||
if (headerAndDataLength < HEADER_SIZE) {
|
||||
@throw [TTransportException exceptionWithReason:@"Framed transport buffer has no header"];
|
||||
UInt32 got = [self readAvail:outBuffer offset:outBufferOffset maxLength:length error:error];
|
||||
if (got != length) {
|
||||
|
||||
// Report underflow only if readAvail didn't report error already
|
||||
if (error && !*error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:nil];
|
||||
}
|
||||
|
||||
size_t dataLength = headerAndDataLength - HEADER_SIZE;
|
||||
uint8_t i32rd[HEADER_SIZE];
|
||||
i32rd[0] = (uint8_t)(0xff & (dataLength >> 24));
|
||||
i32rd[1] = (uint8_t)(0xff & (dataLength >> 16));
|
||||
i32rd[2] = (uint8_t)(0xff & (dataLength >> 8));
|
||||
i32rd[3] = (uint8_t)(0xff & (dataLength));
|
||||
return NO;
|
||||
}
|
||||
|
||||
// should we make a copy of the writeBuffer instead? Better for threaded operations!
|
||||
[writeBuffer replaceBytesInRange:NSMakeRange(0, HEADER_SIZE) withBytes:i32rd length:HEADER_SIZE];
|
||||
[mTransport write:[writeBuffer mutableBytes] offset:0 length:headerAndDataLength];
|
||||
[mTransport flush];
|
||||
|
||||
// reuse old memory buffer
|
||||
[writeBuffer setLength:0];
|
||||
[writeBuffer appendBytes:dummy_header length:HEADER_SIZE];
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length
|
||||
-(UInt32) readAvail:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset maxLength:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[writeBuffer appendBytes:data+offset length:length];
|
||||
UInt32 got = 0;
|
||||
while (got < length) {
|
||||
|
||||
NSUInteger avail = _readBuffer.length - _readOffset;
|
||||
if (avail == 0) {
|
||||
if (![self readFrame:error]) {
|
||||
return 0;
|
||||
}
|
||||
avail = _readBuffer.length;
|
||||
}
|
||||
|
||||
NSRange range;
|
||||
range.location = _readOffset;
|
||||
range.length = MIN(length - got, avail);
|
||||
|
||||
[_readBuffer getBytes:outBuffer+outBufferOffset+got range:range];
|
||||
_readOffset += range.length;
|
||||
got += range.length;
|
||||
}
|
||||
|
||||
return got;
|
||||
}
|
||||
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length
|
||||
-(BOOL) readFrame:(NSError **)error
|
||||
{
|
||||
if (readBuffer == nil) {
|
||||
[self readFrame];
|
||||
}
|
||||
|
||||
if (readBuffer != nil) {
|
||||
size_t bufferLength = [readBuffer length];
|
||||
if (bufferLength - readOffset >= length) {
|
||||
[readBuffer getBytes:buf range:NSMakeRange(readOffset,length)]; // copy data
|
||||
readOffset += length;
|
||||
} else {
|
||||
// void the previous readBuffer data and request a new frame
|
||||
[self readFrame];
|
||||
[readBuffer getBytes:buf range:NSMakeRange(0,length)]; // copy data
|
||||
readOffset = length;
|
||||
}
|
||||
}
|
||||
return length;
|
||||
}
|
||||
UInt8 i32rd[HEADER_SIZE];
|
||||
if (![_transport readAll:i32rd offset:0 length:HEADER_SIZE error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
SInt32 size =
|
||||
((i32rd[0] & 0xff) << 24) |
|
||||
((i32rd[1] & 0xff) << 16) |
|
||||
((i32rd[2] & 0xff) << 8) |
|
||||
((i32rd[3] & 0xff));
|
||||
|
||||
if (_readBuffer == nil) {
|
||||
|
||||
_readBuffer = [NSMutableData dataWithLength:size];
|
||||
|
||||
}
|
||||
else {
|
||||
|
||||
SInt32 len = (SInt32)_readBuffer.length;
|
||||
if (len >= size) {
|
||||
|
||||
_readBuffer.length = size;
|
||||
|
||||
}
|
||||
else {
|
||||
|
||||
// increase length of data buffer
|
||||
[_readBuffer increaseLengthBy:size-len];
|
||||
|
||||
- (void)readFrame
|
||||
{
|
||||
uint8_t i32rd[HEADER_SIZE];
|
||||
[mTransport readAll: i32rd offset: 0 length: HEADER_SIZE];
|
||||
int32_t headerValue =
|
||||
((i32rd[0] & 0xff) << 24) |
|
||||
((i32rd[1] & 0xff) << 16) |
|
||||
((i32rd[2] & 0xff) << 8) |
|
||||
((i32rd[3] & 0xff));
|
||||
if (headerValue < 0) {
|
||||
NSString *reason = [NSString stringWithFormat:
|
||||
@"Frame header reports negative frame size: %"PRId32,
|
||||
headerValue];
|
||||
@throw [TTransportException exceptionWithReason:reason];
|
||||
}
|
||||
|
||||
/* Cast should be safe:
|
||||
* Have verified headerValue non-negative and of lesser or equal bitwidth to size_t. */
|
||||
size_t frameSize = (size_t)headerValue;
|
||||
[self ensureReadBufferHasLength:frameSize];
|
||||
}
|
||||
|
||||
[mTransport readAll:[readBuffer mutableBytes] offset:0 length:frameSize];
|
||||
}
|
||||
// copy into internal memory buffer
|
||||
if (![_transport readAll:_readBuffer.mutableBytes offset:0 length:size error:error]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
- (void)ensureReadBufferHasLength:(size_t)length
|
||||
{
|
||||
if (readBuffer == nil) {
|
||||
readBuffer = [[NSMutableData alloc] initWithLength:length];
|
||||
} else {
|
||||
size_t currentLength = [readBuffer length];
|
||||
BOOL isTooLong = (currentLength >= length);
|
||||
if (isTooLong) {
|
||||
[readBuffer setLength:length];
|
||||
} else {
|
||||
size_t lengthToAdd = length - currentLength;
|
||||
[readBuffer increaseLengthBy:lengthToAdd];
|
||||
}
|
||||
}
|
||||
return YES;
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -1,161 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "THTTPClient.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
@implementation THTTPClient
|
||||
|
||||
|
||||
- (void) setupRequest
|
||||
{
|
||||
if (mRequest != nil) {
|
||||
[mRequest release_stub];
|
||||
}
|
||||
|
||||
// set up our request object that we'll use for each request
|
||||
mRequest = [[NSMutableURLRequest alloc] initWithURL: mURL];
|
||||
[mRequest setHTTPMethod: @"POST"];
|
||||
[mRequest setValue: @"application/x-thrift" forHTTPHeaderField: @"Content-Type"];
|
||||
[mRequest setValue: @"application/x-thrift" forHTTPHeaderField: @"Accept"];
|
||||
|
||||
NSString * userAgent = mUserAgent;
|
||||
if (!userAgent) {
|
||||
userAgent = @"Cocoa/THTTPClient";
|
||||
}
|
||||
[mRequest setValue: userAgent forHTTPHeaderField: @"User-Agent"];
|
||||
|
||||
[mRequest setCachePolicy: NSURLRequestReloadIgnoringCacheData];
|
||||
if (mTimeout) {
|
||||
[mRequest setTimeoutInterval: mTimeout];
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
- (id) initWithURL: (NSURL *) aURL
|
||||
{
|
||||
return [self initWithURL: aURL
|
||||
userAgent: nil
|
||||
timeout: 0];
|
||||
}
|
||||
|
||||
|
||||
- (id) initWithURL: (NSURL *) aURL
|
||||
userAgent: (NSString *) userAgent
|
||||
timeout: (int) timeout
|
||||
{
|
||||
self = [super init];
|
||||
if (!self) {
|
||||
return nil;
|
||||
}
|
||||
|
||||
mTimeout = timeout;
|
||||
if (userAgent) {
|
||||
mUserAgent = [userAgent retain_stub];
|
||||
}
|
||||
mURL = [aURL retain_stub];
|
||||
|
||||
[self setupRequest];
|
||||
|
||||
// create our request data buffer
|
||||
mRequestData = [[NSMutableData alloc] initWithCapacity: 1024];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
|
||||
- (void) setURL: (NSURL *) aURL
|
||||
{
|
||||
[aURL retain_stub];
|
||||
[mURL release_stub];
|
||||
mURL = aURL;
|
||||
|
||||
[self setupRequest];
|
||||
}
|
||||
|
||||
|
||||
- (void) dealloc
|
||||
{
|
||||
[mURL release_stub];
|
||||
[mUserAgent release_stub];
|
||||
[mRequest release_stub];
|
||||
[mRequestData release_stub];
|
||||
[mResponseData release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
|
||||
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length
|
||||
{
|
||||
NSRange r;
|
||||
r.location = mResponseDataOffset;
|
||||
r.length = length;
|
||||
|
||||
[mResponseData getBytes: buf+offset range: r];
|
||||
mResponseDataOffset += length;
|
||||
|
||||
return length;
|
||||
}
|
||||
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length
|
||||
{
|
||||
[mRequestData appendBytes: data+offset length: length];
|
||||
}
|
||||
|
||||
|
||||
- (void) flush
|
||||
{
|
||||
[mRequest setHTTPBody: mRequestData]; // not sure if it copies the data
|
||||
|
||||
// make the HTTP request
|
||||
NSURLResponse * response;
|
||||
NSError * error;
|
||||
NSData * responseData =
|
||||
[NSURLConnection sendSynchronousRequest: mRequest returningResponse: &response error: &error];
|
||||
|
||||
[mRequestData setLength: 0];
|
||||
|
||||
if (responseData == nil) {
|
||||
@throw [TTransportException exceptionWithName: @"TTransportException"
|
||||
reason: @"Could not make HTTP request"
|
||||
error: error];
|
||||
}
|
||||
if (![response isKindOfClass: [NSHTTPURLResponse class]]) {
|
||||
@throw [TTransportException exceptionWithName: @"TTransportException"
|
||||
reason: [NSString stringWithFormat: @"Unexpected NSURLResponse type: %@",
|
||||
NSStringFromClass([response class])]];
|
||||
}
|
||||
|
||||
NSHTTPURLResponse * httpResponse = (NSHTTPURLResponse *) response;
|
||||
if ([httpResponse statusCode] != 200) {
|
||||
@throw [TTransportException exceptionWithName: @"TTransportException"
|
||||
reason: [NSString stringWithFormat: @"Bad response from HTTP server: %ld",
|
||||
(long)[httpResponse statusCode]]];
|
||||
}
|
||||
|
||||
// phew!
|
||||
[mResponseData release_stub];
|
||||
mResponseData = [responseData retain_stub];
|
||||
mResponseDataOffset = 0;
|
||||
}
|
||||
|
||||
|
||||
@end
|
47
lib/cocoa/src/transport/THTTPSessionTransport.h
Normal file
47
lib/cocoa/src/transport/THTTPSessionTransport.h
Normal file
@ -0,0 +1,47 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TAsyncTransport.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
typedef NSError *__nullable (^THTTPSessionTransportResponseValidateBlock) (NSHTTPURLResponse *response, NSData *responseData);
|
||||
|
||||
|
||||
@interface THTTPSessionTransportFactory : NSObject<TAsyncTransportFactory>
|
||||
|
||||
@property (strong, nonatomic) THTTPSessionTransportResponseValidateBlock responseValidate;
|
||||
|
||||
+(void) setupDefaultsForSessionConfiguration:(NSURLSessionConfiguration *)config
|
||||
withProtocolName:(NSString *)protocolName;
|
||||
|
||||
-(id) initWithSession:(NSURLSession *)session
|
||||
URL:(NSURL *)aURL;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@interface THTTPSessionTransport : NSObject <TAsyncTransport>
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
268
lib/cocoa/src/transport/THTTPSessionTransport.m
Normal file
268
lib/cocoa/src/transport/THTTPSessionTransport.m
Normal file
@ -0,0 +1,268 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "THTTPSessionTransport.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
@interface THTTPSessionTransportFactory ()
|
||||
|
||||
@property (strong, nonatomic) NSURLSession *session;
|
||||
@property (strong, nonatomic) NSURL *url;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@interface THTTPSessionTransport ()
|
||||
|
||||
@property (strong, nonatomic) THTTPSessionTransportFactory *factory;
|
||||
@property (strong, nonatomic) NSMutableData *requestData;
|
||||
@property (strong, nonatomic) NSData *responseData;
|
||||
@property (assign, nonatomic) NSUInteger responseDataOffset;
|
||||
|
||||
-(instancetype) initWithFactory:(THTTPSessionTransportFactory *)factory;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation THTTPSessionTransportFactory
|
||||
|
||||
+(void) setupDefaultsForSessionConfiguration:(NSURLSessionConfiguration *)config withProtocolName:(NSString *)protocolName
|
||||
{
|
||||
NSString *thriftContentType = @"application/x-thrift";
|
||||
if (protocolName.length) {
|
||||
thriftContentType = [thriftContentType stringByAppendingFormat:@"; p=%@", protocolName];
|
||||
}
|
||||
|
||||
config.requestCachePolicy = NSURLRequestReloadIgnoringCacheData;
|
||||
config.HTTPShouldUsePipelining = YES;
|
||||
config.HTTPShouldSetCookies = NO;
|
||||
config.URLCache = nil;
|
||||
config.HTTPAdditionalHeaders = @{@"Content-Type":thriftContentType,
|
||||
@"Accept":thriftContentType,
|
||||
@"User-Agent":@"Thrift/Cocoa (Session)"};
|
||||
}
|
||||
|
||||
|
||||
-(id) initWithSession:(NSURLSession *)session URL:(NSURL *)url
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_session = session;
|
||||
_url = url;
|
||||
}
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
-(id<TAsyncTransport>) newTransport
|
||||
{
|
||||
return [[THTTPSessionTransport alloc] initWithFactory:self];
|
||||
}
|
||||
|
||||
-(NSURLSessionDataTask *) taskWithRequest:(NSURLRequest *)request
|
||||
completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler
|
||||
error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
NSURLSessionDataTask *newTask = [_session dataTaskWithRequest:request completionHandler:completionHandler];
|
||||
if (!newTask) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{NSLocalizedDescriptionKey:@"Failed to create session data task"}];
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
|
||||
return newTask;
|
||||
}
|
||||
|
||||
-(NSError *) validateResponse:(NSHTTPURLResponse *)response data:(NSData *)data
|
||||
{
|
||||
if (_responseValidate) {
|
||||
return _responseValidate(response, data);
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
||||
@implementation THTTPSessionTransport
|
||||
|
||||
-(instancetype) initWithFactory:(THTTPSessionTransportFactory *)factory
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_factory = factory;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
-(BOOL) readAll:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
UInt32 got = [self readAvail:outBuffer offset:outBufferOffset maxLength:length error:error];
|
||||
if (got != length) {
|
||||
|
||||
// Report underflow only if readAvail didn't report error already
|
||||
if (error && !*error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:nil];
|
||||
}
|
||||
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(UInt32) readAvail:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset maxLength:(UInt32)maxLength error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
NSUInteger avail = _responseData.length - _responseDataOffset;
|
||||
|
||||
NSRange range;
|
||||
range.location = _responseDataOffset;
|
||||
range.length = MIN(maxLength, avail);
|
||||
|
||||
[_responseData getBytes:outBuffer+outBufferOffset range:range];
|
||||
_responseDataOffset += range.length;
|
||||
|
||||
return (UInt32)range.length;
|
||||
}
|
||||
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
if (!_requestData) {
|
||||
_requestData = [NSMutableData dataWithCapacity:256];
|
||||
}
|
||||
|
||||
[_requestData appendBytes:data+offset length:length];
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(void) flushWithCompletion:(TAsyncCompletionBlock)completed failure:(TAsyncFailureBlock)failure
|
||||
{
|
||||
NSError *error;
|
||||
|
||||
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:_factory.url];
|
||||
request.HTTPMethod = @"POST";
|
||||
request.HTTPBody = _requestData;
|
||||
|
||||
_requestData = nil;
|
||||
|
||||
NSURLSessionDataTask *task = [_factory taskWithRequest:request completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
|
||||
|
||||
// Check response type
|
||||
if (!error && ![response isKindOfClass:NSHTTPURLResponse.class]) {
|
||||
|
||||
error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{TTransportErrorHttpErrorKey: @(THttpTransportErrorInvalidResponse)}];
|
||||
|
||||
}
|
||||
|
||||
// Check status code
|
||||
NSHTTPURLResponse *httpResponse = (id)response;
|
||||
if (!error && httpResponse.statusCode != 200) {
|
||||
|
||||
THttpTransportError code;
|
||||
if (httpResponse.statusCode == 401) {
|
||||
code = THttpTransportErrorAuthentication;
|
||||
}
|
||||
else {
|
||||
code = THttpTransportErrorInvalidStatus;
|
||||
}
|
||||
|
||||
error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{TTransportErrorHttpErrorKey: @(code),
|
||||
@"statusCode":@(httpResponse.statusCode)}];
|
||||
}
|
||||
|
||||
// Allow factory to check
|
||||
if (!error) {
|
||||
error = [_factory validateResponse:httpResponse data:data];
|
||||
}
|
||||
|
||||
_responseDataOffset = 0;
|
||||
|
||||
if (error) {
|
||||
|
||||
_responseData = nil;
|
||||
|
||||
failure(error);
|
||||
|
||||
}
|
||||
else {
|
||||
|
||||
if (data == nil) {
|
||||
data = [NSData data];
|
||||
}
|
||||
|
||||
_responseData = data;
|
||||
|
||||
completed(self);
|
||||
}
|
||||
|
||||
} error:&error];
|
||||
|
||||
if (!task) {
|
||||
failure(error);
|
||||
return;
|
||||
}
|
||||
|
||||
[task resume];
|
||||
}
|
||||
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error
|
||||
{
|
||||
dispatch_semaphore_t completed = dispatch_semaphore_create(0);
|
||||
|
||||
__block BOOL result;
|
||||
__block NSError *internalError;
|
||||
|
||||
[self flushWithCompletion:^(id < TAsyncTransport > transport) {
|
||||
|
||||
result = YES;
|
||||
|
||||
dispatch_semaphore_signal(completed);
|
||||
|
||||
} failure:^(NSError *error) {
|
||||
|
||||
internalError = error;
|
||||
|
||||
result = NO;
|
||||
|
||||
dispatch_semaphore_signal(completed);
|
||||
|
||||
}];
|
||||
|
||||
dispatch_semaphore_wait(completed, DISPATCH_TIME_FOREVER);
|
||||
|
||||
if (error) {
|
||||
*error = internalError;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
@end
|
@ -18,17 +18,22 @@
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TException : NSException {
|
||||
}
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name;
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name
|
||||
reason: (NSString *) reason;
|
||||
@interface THTTPTransport : NSObject <TTransport>
|
||||
|
||||
+ (id) exceptionWithName: (NSString *) name
|
||||
reason: (NSString *) reason
|
||||
error: (NSError *) error;
|
||||
-(id) initWithURL:(NSURL *)aURL;
|
||||
|
||||
-(id) initWithURL:(NSURL *)aURL
|
||||
userAgent:(nullable NSString *)userAgent
|
||||
timeout:(int)timeout;
|
||||
|
||||
-(void) setURL:(NSURL *)aURL;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
182
lib/cocoa/src/transport/THTTPTransport.m
Normal file
182
lib/cocoa/src/transport/THTTPTransport.m
Normal file
@ -0,0 +1,182 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "THTTPTransport.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
@interface THTTPTransport ()
|
||||
|
||||
@property (strong, nonatomic) NSURL *url;
|
||||
@property (strong, nonatomic) NSMutableURLRequest *request;
|
||||
@property (strong, nonatomic) NSMutableData *requestData;
|
||||
@property (strong, nonatomic) NSData *responseData;
|
||||
@property (assign, nonatomic) NSUInteger responseDataOffset;
|
||||
@property (strong, nonatomic) NSString *userAgent;
|
||||
@property (assign, nonatomic) NSTimeInterval timeout;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation THTTPTransport
|
||||
|
||||
-(void) setupRequest
|
||||
{
|
||||
// set up our request object that we'll use for each request
|
||||
_request = [[NSMutableURLRequest alloc] initWithURL:_url];
|
||||
[_request setHTTPMethod:@"POST"];
|
||||
[_request setValue:@"application/x-thrift" forHTTPHeaderField:@"Content-Type"];
|
||||
[_request setValue:@"application/x-thrift" forHTTPHeaderField:@"Accept"];
|
||||
|
||||
NSString *userAgent = _userAgent;
|
||||
if (!userAgent) {
|
||||
userAgent = @"Thrift/Cocoa";
|
||||
}
|
||||
[_request setValue:userAgent forHTTPHeaderField:@"User-Agent"];
|
||||
|
||||
[_request setCachePolicy:NSURLRequestReloadIgnoringCacheData];
|
||||
if (_timeout) {
|
||||
[_request setTimeoutInterval:_timeout];
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
-(id) initWithURL:(NSURL *)aURL
|
||||
{
|
||||
return [self initWithURL:aURL
|
||||
userAgent:nil
|
||||
timeout:0];
|
||||
}
|
||||
|
||||
|
||||
-(id) initWithURL:(NSURL *)aURL
|
||||
userAgent:(NSString *)aUserAgent
|
||||
timeout:(int)aTimeout
|
||||
{
|
||||
self = [super init];
|
||||
if (!self) {
|
||||
return nil;
|
||||
}
|
||||
|
||||
_timeout = aTimeout;
|
||||
_userAgent = aUserAgent;
|
||||
_url = aURL;
|
||||
|
||||
[self setupRequest];
|
||||
|
||||
// create our request data buffer
|
||||
_requestData = [[NSMutableData alloc] initWithCapacity:1024];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
-(void) setURL:(NSURL *)aURL
|
||||
{
|
||||
_url = aURL;
|
||||
|
||||
[self setupRequest];
|
||||
}
|
||||
|
||||
-(BOOL) readAll:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
UInt32 got = [self readAvail:outBuffer offset:outBufferOffset maxLength:length error:error];
|
||||
if (got != length) {
|
||||
|
||||
// Report underflow only if readAvail didn't report error already
|
||||
if (error && !*error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:nil];
|
||||
}
|
||||
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(UInt32) readAvail:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset maxLength:(UInt32)maxLength error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
NSUInteger avail = _responseData.length - _responseDataOffset;
|
||||
|
||||
NSRange range;
|
||||
range.location = _responseDataOffset;
|
||||
range.length = MIN(maxLength, avail);
|
||||
|
||||
[_responseData getBytes:outBuffer+outBufferOffset range:range];
|
||||
_responseDataOffset += range.length;
|
||||
|
||||
return (UInt32)range.length;
|
||||
}
|
||||
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[_requestData appendBytes:data+offset length:length];
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[_request setHTTPBody:_requestData];
|
||||
|
||||
_responseDataOffset = 0;
|
||||
|
||||
// make the HTTP request
|
||||
NSURLResponse *response;
|
||||
_responseData = [NSURLConnection sendSynchronousRequest:_request returningResponse:&response error:error];
|
||||
if (!_responseData) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
[_requestData setLength:0];
|
||||
|
||||
if (![response isKindOfClass:NSHTTPURLResponse.class]) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{TTransportErrorHttpErrorKey: @(THttpTransportErrorInvalidResponse)}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response;
|
||||
if ([httpResponse statusCode] != 200) {
|
||||
if (error) {
|
||||
|
||||
THttpTransportError code;
|
||||
if (httpResponse.statusCode == 401) {
|
||||
code = THttpTransportErrorAuthentication;
|
||||
}
|
||||
else {
|
||||
code = THttpTransportErrorInvalidStatus;
|
||||
}
|
||||
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorUnknown
|
||||
userInfo:@{TTransportErrorHttpErrorKey: @(code),
|
||||
@"statusCode":@(httpResponse.statusCode)}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
@end
|
@ -20,10 +20,18 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TMemoryBuffer : NSObject <TTransport> {
|
||||
NSMutableData *mBuffer;
|
||||
NSUInteger mOffset;
|
||||
}
|
||||
- (id)initWithData:(NSData *)data;
|
||||
- (NSData *)getBuffer;
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@interface TMemoryBuffer : NSObject <TTransport>
|
||||
|
||||
-(NSData *) buffer;
|
||||
|
||||
-(id) initWithData:(NSData *)data;
|
||||
|
||||
-(id) initWithDataNoCopy:(NSMutableData *)data;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -18,57 +18,104 @@
|
||||
*/
|
||||
|
||||
#import "TMemoryBuffer.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
#define GARBAGE_BUFFER_SIZE 4096 // 4KiB
|
||||
|
||||
@implementation TMemoryBuffer
|
||||
- (id)init {
|
||||
if ((self = [super init])) {
|
||||
mBuffer = [[NSMutableData alloc] init];
|
||||
mOffset = 0;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (id)initWithData:(NSData *)data {
|
||||
if ((self = [super init])) {
|
||||
mBuffer = [data mutableCopy];
|
||||
mOffset = 0;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
@interface TMemoryBuffer ()
|
||||
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length
|
||||
{
|
||||
if ([mBuffer length] - mOffset < length) {
|
||||
@throw [TTransportException exceptionWithReason:@"Not enough bytes remain in buffer"];
|
||||
}
|
||||
[mBuffer getBytes:buf range:NSMakeRange(mOffset, length)];
|
||||
mOffset += length;
|
||||
if (mOffset >= GARBAGE_BUFFER_SIZE) {
|
||||
[mBuffer replaceBytesInRange:NSMakeRange(0, mOffset) withBytes:NULL length:0];
|
||||
mOffset = 0;
|
||||
}
|
||||
return length;
|
||||
}
|
||||
@property(strong, nonatomic) NSMutableData *buffer;
|
||||
@property(assign, nonatomic) UInt32 bufferOffset;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TMemoryBuffer
|
||||
|
||||
-(id) init
|
||||
{
|
||||
if ((self = [super init])) {
|
||||
_buffer = [NSMutableData new];
|
||||
_bufferOffset = 0;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
-(id) initWithData:(NSData *)data
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_buffer = [data mutableCopy];
|
||||
_bufferOffset = 0;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
-(id) initWithDataNoCopy:(NSMutableData *)data
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_buffer = data;
|
||||
_bufferOffset = 0;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
-(BOOL) readAll:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
UInt32 got = [self readAvail:outBuffer offset:outBufferOffset maxLength:length error:error];
|
||||
if (got != length) {
|
||||
|
||||
// Report underflow only if readAvail didn't report error already
|
||||
if (error && !*error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:nil];
|
||||
}
|
||||
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(UInt32) readAvail:(UInt8 *)outBuffer offset:(UInt32)outBufferOffset maxLength:(UInt32)maxLength error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
UInt32 avail = (UInt32)_buffer.length - _bufferOffset;
|
||||
if (avail == 0) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
NSRange range;
|
||||
range.location = _bufferOffset;
|
||||
range.length = MIN(maxLength, avail);
|
||||
|
||||
[_buffer getBytes:outBuffer + outBufferOffset range:range];
|
||||
_bufferOffset += range.length;
|
||||
|
||||
if (_bufferOffset >= GARBAGE_BUFFER_SIZE) {
|
||||
[_buffer replaceBytesInRange:NSMakeRange(0, _bufferOffset) withBytes:NULL length:0];
|
||||
_bufferOffset = 0;
|
||||
}
|
||||
|
||||
return (UInt32)range.length;
|
||||
}
|
||||
|
||||
-(BOOL) write:(const UInt8 *)inBuffer offset:(UInt32)inBufferOffset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
[_buffer appendBytes:inBuffer + inBufferOffset length:length];
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(NSData *) buffer
|
||||
{
|
||||
return _buffer;
|
||||
}
|
||||
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length
|
||||
{
|
||||
[mBuffer appendBytes:data+offset length:length];
|
||||
}
|
||||
|
||||
- (void)flush {
|
||||
// noop
|
||||
}
|
||||
|
||||
- (NSData *)getBuffer {
|
||||
return [[mBuffer copy] autorelease_stub];
|
||||
}
|
||||
|
||||
- (void)dealloc {
|
||||
[mBuffer release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
@end
|
||||
|
@ -21,15 +21,18 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TNSFileHandleTransport : NSObject <TTransport> {
|
||||
NSFileHandle * mInputFileHandle;
|
||||
NSFileHandle * mOutputFileHandle;
|
||||
}
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
- (id) initWithFileHandle: (NSFileHandle *) fileHandle;
|
||||
|
||||
- (id) initWithInputFileHandle: (NSFileHandle *) inputFileHandle
|
||||
outputFileHandle: (NSFileHandle *) outputFileHandle;
|
||||
@interface TNSFileHandleTransport : NSObject <TTransport>
|
||||
|
||||
-(id) initWithFileHandle:(NSFileHandle *)fileHandle;
|
||||
|
||||
-(id) initWithInputFileHandle:(NSFileHandle *)inputFileHandle
|
||||
outputFileHandle:(NSFileHandle *)outputFileHandle;
|
||||
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -19,75 +19,100 @@
|
||||
|
||||
|
||||
#import "TNSFileHandleTransport.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
@interface TNSFileHandleTransport ()
|
||||
|
||||
@property(strong, nonatomic) NSFileHandle *inputFileHandle;
|
||||
@property(strong, nonatomic) NSFileHandle *outputFileHandle;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TNSFileHandleTransport
|
||||
|
||||
- (id) initWithFileHandle: (NSFileHandle *) fileHandle
|
||||
-(id) initWithFileHandle:(NSFileHandle *)fileHandle
|
||||
{
|
||||
return [self initWithInputFileHandle: fileHandle
|
||||
outputFileHandle: fileHandle];
|
||||
return [self initWithInputFileHandle:fileHandle
|
||||
outputFileHandle:fileHandle];
|
||||
}
|
||||
|
||||
|
||||
- (id) initWithInputFileHandle: (NSFileHandle *) inputFileHandle
|
||||
outputFileHandle: (NSFileHandle *) outputFileHandle
|
||||
-(id) initWithInputFileHandle:(NSFileHandle *)aInputFileHandle
|
||||
outputFileHandle:(NSFileHandle *)aOutputFileHandle
|
||||
{
|
||||
self = [super init];
|
||||
|
||||
mInputFileHandle = [inputFileHandle retain_stub];
|
||||
mOutputFileHandle = [outputFileHandle retain_stub];
|
||||
|
||||
if (self) {
|
||||
_inputFileHandle = aInputFileHandle;
|
||||
_outputFileHandle = aOutputFileHandle;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
|
||||
- (void) dealloc {
|
||||
[mInputFileHandle release_stub];
|
||||
[mOutputFileHandle release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
|
||||
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length
|
||||
-(BOOL) readAll:(UInt8 *)buf offset:(UInt32)off length:(UInt32)len error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
size_t totalBytesRead = 0;
|
||||
while (totalBytesRead < length) {
|
||||
NSData * data = [mInputFileHandle readDataOfLength: length-totalBytesRead];
|
||||
if ([data length] == 0) {
|
||||
@throw [TTransportException exceptionWithName: @"TTransportException"
|
||||
reason: @"Cannot read. No more data."];
|
||||
UInt32 got = 0;
|
||||
while (got < len) {
|
||||
|
||||
NSData *d = [_inputFileHandle readDataOfLength:len-got];
|
||||
if (d.length == 0) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:nil];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
[data getBytes: buf+totalBytesRead];
|
||||
totalBytesRead += [data length];
|
||||
|
||||
[d getBytes:buf+got length:d.length];
|
||||
got += d.length;
|
||||
}
|
||||
return totalBytesRead;
|
||||
return YES;
|
||||
}
|
||||
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length
|
||||
-(UInt32) readAvail:(UInt8 *)buf offset:(UInt32)off maxLength:(UInt32)len error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
const void *pos = data + offset;
|
||||
NSData * dataObject = [[NSData alloc] initWithBytesNoCopy: (void *)pos
|
||||
length: length
|
||||
freeWhenDone: NO];
|
||||
UInt32 got = 0;
|
||||
while (got < len) {
|
||||
|
||||
NSData *d = [_inputFileHandle readDataOfLength:len-got];
|
||||
if (d.length == 0) {
|
||||
break;
|
||||
}
|
||||
|
||||
[d getBytes:buf+got length:d.length];
|
||||
got += d.length;
|
||||
}
|
||||
return got;
|
||||
}
|
||||
|
||||
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
void *pos = (void *)data + offset;
|
||||
|
||||
@try {
|
||||
[mOutputFileHandle writeData: dataObject];
|
||||
} @catch (NSException * e) {
|
||||
@throw [TTransportException exceptionWithName: @"TTransportException"
|
||||
reason: [NSString stringWithFormat: @"%s: Unable to write data: %@", __PRETTY_FUNCTION__, e]];
|
||||
[_outputFileHandle writeData:[NSData dataWithBytesNoCopy:pos length:length freeWhenDone:NO]];
|
||||
}
|
||||
@catch (NSException *e) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorNotOpen
|
||||
userInfo:@{}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
[dataObject release_stub];
|
||||
return YES;
|
||||
}
|
||||
|
||||
|
||||
- (void) flush
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error
|
||||
{
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -20,21 +20,24 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
|
||||
@interface TNSStreamTransport : NSObject <TTransport> {
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
}
|
||||
|
||||
@property (nonatomic, strong) NSInputStream * mInput;
|
||||
@property (nonatomic, strong) NSOutputStream * mOutput;
|
||||
@interface TNSStreamTransport : NSObject <TTransport>
|
||||
|
||||
- (id) initWithInputStream: (NSInputStream *) input
|
||||
outputStream: (NSOutputStream *) output;
|
||||
@property (strong, nonatomic) NSInputStream *input;
|
||||
@property (strong, nonatomic) NSOutputStream *output;
|
||||
|
||||
- (id) initWithInputStream: (NSInputStream *) input;
|
||||
-(id) initWithInputStream:(nullable NSInputStream *)input
|
||||
outputStream:(nullable NSOutputStream *)output;
|
||||
|
||||
- (id) initWithOutputStream: (NSOutputStream *) output;
|
||||
-(id) initWithInputStream:(NSInputStream *)input;
|
||||
|
||||
-(id) initWithOutputStream:(NSOutputStream *)output;
|
||||
|
||||
-(void) close;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -18,79 +18,136 @@
|
||||
*/
|
||||
|
||||
#import "TNSStreamTransport.h"
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
@interface TNSStreamTransport ()
|
||||
@end
|
||||
|
||||
|
||||
@implementation TNSStreamTransport
|
||||
|
||||
- (id) initWithInputStream: (NSInputStream *) input
|
||||
outputStream: (NSOutputStream *) output
|
||||
-(id) initWithInputStream:(NSInputStream *)input
|
||||
outputStream:(NSOutputStream *)output
|
||||
{
|
||||
self = [super init];
|
||||
self.mInput = [input retain_stub];
|
||||
self.mOutput = [output retain_stub];
|
||||
if (self) {
|
||||
_input = input;
|
||||
_output = output;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (id) initWithInputStream: (NSInputStream *) input
|
||||
-(id) initWithInputStream:(NSInputStream *)input
|
||||
{
|
||||
return [self initWithInputStream: input outputStream: nil];
|
||||
return [self initWithInputStream:input outputStream:nil];
|
||||
}
|
||||
|
||||
- (id) initWithOutputStream: (NSOutputStream *) output
|
||||
-(id) initWithOutputStream:(NSOutputStream *)output
|
||||
{
|
||||
return [self initWithInputStream: nil outputStream: output];
|
||||
return [self initWithInputStream:nil outputStream:output];
|
||||
}
|
||||
|
||||
- (void) dealloc
|
||||
-(void) dealloc
|
||||
{
|
||||
[self.mInput release_stub];
|
||||
[self.mOutput release_stub];
|
||||
[super dealloc_stub];
|
||||
[self close];
|
||||
}
|
||||
|
||||
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length
|
||||
-(BOOL) readAll:(UInt8 *)buf offset:(UInt32)off length:(UInt32)len error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
size_t totalBytesRead = 0;
|
||||
ssize_t bytesRead = 0;
|
||||
while (totalBytesRead < length) {
|
||||
bytesRead = [self.mInput read: buf+offset+totalBytesRead maxLength: length-totalBytesRead];
|
||||
UInt32 got = 0;
|
||||
while (got < len) {
|
||||
|
||||
BOOL encounteredErrorOrEOF = (bytesRead <= 0);
|
||||
if (encounteredErrorOrEOF) {
|
||||
@throw [TTransportException exceptionWithReason: @"Cannot read. Remote side has closed."];
|
||||
} else {
|
||||
/* bytesRead is guaranteed to be positive and within the range representable by size_t. */
|
||||
totalBytesRead += (size_t)bytesRead;
|
||||
UInt32 read = (UInt32)[_input read:buf+off+got maxLength:len-got];
|
||||
if (read <= 0) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorNotOpen
|
||||
userInfo:@{}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
got += read;
|
||||
}
|
||||
return totalBytesRead;
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length
|
||||
-(UInt32) readAvail:(UInt8 *)buf offset:(UInt32)off maxLength:(UInt32)len error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
size_t totalBytesWritten = 0;
|
||||
ssize_t bytesWritten = 0;
|
||||
while (totalBytesWritten < length) {
|
||||
bytesWritten = [self.mOutput write: data+offset+totalBytesWritten maxLength: length-totalBytesWritten];
|
||||
if (bytesWritten < 0) {
|
||||
@throw [TTransportException exceptionWithReason: @"Error writing to transport output stream."
|
||||
error: [self.mOutput streamError]];
|
||||
} else if (bytesWritten == 0) {
|
||||
@throw [TTransportException exceptionWithReason: @"End of output stream."];
|
||||
} else {
|
||||
/* bytesWritten is guaranteed to be positive and within the range representable by size_t. */
|
||||
totalBytesWritten += (size_t)bytesWritten;
|
||||
UInt32 got = 0;
|
||||
while (got < len) {
|
||||
|
||||
UInt32 read = (UInt32)[_input read:buf+off+got maxLength:len-got];
|
||||
if (read <= 0) {
|
||||
break;
|
||||
}
|
||||
|
||||
got += read;
|
||||
}
|
||||
|
||||
return got;
|
||||
}
|
||||
|
||||
- (void) flush
|
||||
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error
|
||||
{
|
||||
// no flush for you!
|
||||
int got = 0;
|
||||
NSInteger total = 0;
|
||||
while (got < length) {
|
||||
|
||||
total = [_output write:data+offset+got maxLength:length-got];
|
||||
if (total == -1) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorNotOpen
|
||||
userInfo:@{}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
else if (total == 0) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TTransportErrorDomain
|
||||
code:TTransportErrorEndOfFile
|
||||
userInfo:@{}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
got += total;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error
|
||||
{
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(void) close
|
||||
{
|
||||
NSInputStream *input = self.input;
|
||||
if (input) {
|
||||
// Close and reset inputstream
|
||||
CFReadStreamSetProperty((__bridge CFReadStreamRef)(input), kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
[input setDelegate:nil];
|
||||
[input close];
|
||||
[input removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
input = nil;
|
||||
}
|
||||
|
||||
NSOutputStream *output = self.output;
|
||||
if (output) {
|
||||
// Close and reset outputstream
|
||||
CFWriteStreamSetProperty((__bridge CFWriteStreamRef)(output), kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
[output setDelegate:nil];
|
||||
[output close];
|
||||
[output removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
output = nil;
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -1,261 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
#import <Foundation/Foundation.h>
|
||||
#import <CoreFoundation/CoreFoundation.h>
|
||||
#import "TSSLSocketClient.h"
|
||||
#import "TSSLSocketException.h"
|
||||
#import "TObjective-C.h"
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
#include <netdb.h>
|
||||
|
||||
#if !TARGET_OS_IPHONE
|
||||
#import <CoreServices/CoreServices.h>
|
||||
#else
|
||||
#import <CFNetwork/CFNetwork.h>
|
||||
#endif
|
||||
|
||||
@implementation TSSLSocketClient
|
||||
|
||||
- (id) initWithHostname: (NSString *) hostname
|
||||
port: (int) port
|
||||
{
|
||||
sslHostname = hostname;
|
||||
CFReadStreamRef readStream = NULL;
|
||||
CFWriteStreamRef writeStream = NULL;
|
||||
|
||||
|
||||
/* create a socket structure */
|
||||
struct sockaddr_in pin;
|
||||
struct hostent *hp = NULL;
|
||||
for(int i = 0; i < 10; i++) {
|
||||
|
||||
|
||||
|
||||
if ((hp = gethostbyname([hostname UTF8String])) == NULL) {
|
||||
NSLog(@"failed to resolve hostname %@", hostname);
|
||||
herror("resolv");
|
||||
if(i == 9) {
|
||||
@throw [TSSLSocketException exceptionWithReason: @"failed to resolve hostname"];
|
||||
}
|
||||
[NSThread sleepForTimeInterval:0.2];
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
memset (&pin, 0, sizeof(pin));
|
||||
pin.sin_family = AF_INET;
|
||||
memcpy(&pin.sin_addr, hp->h_addr, sizeof(struct in_addr));
|
||||
pin.sin_port = htons (port);
|
||||
|
||||
/* create the socket */
|
||||
if ((sd = socket (AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1)
|
||||
{
|
||||
NSLog(@"failed to create socket for host %@:%d", hostname, port);
|
||||
@throw [TSSLSocketException exceptionWithReason: @"failed to create socket"];
|
||||
}
|
||||
|
||||
/* open a connection */
|
||||
if (connect (sd, (struct sockaddr *) &pin, sizeof(pin)) == -1)
|
||||
{
|
||||
NSLog(@"failed to create conenct to host %@:%d", hostname, port);
|
||||
@throw [TSSLSocketException exceptionWithReason: @"failed to connect"];
|
||||
}
|
||||
CFStreamCreatePairWithSocket(kCFAllocatorDefault, sd, &readStream, &writeStream);
|
||||
|
||||
CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
|
||||
if (readStream && writeStream) {
|
||||
CFReadStreamSetProperty(readStream,
|
||||
kCFStreamPropertySocketSecurityLevel,
|
||||
kCFStreamSocketSecurityLevelTLSv1);
|
||||
|
||||
NSDictionary *settings =
|
||||
[NSDictionary dictionaryWithObjectsAndKeys:
|
||||
(id)kCFBooleanTrue, (id)kCFStreamSSLValidatesCertificateChain,
|
||||
nil];
|
||||
|
||||
CFReadStreamSetProperty((CFReadStreamRef)readStream,
|
||||
kCFStreamPropertySSLSettings,
|
||||
(CFTypeRef)settings);
|
||||
CFWriteStreamSetProperty((CFWriteStreamRef)writeStream,
|
||||
kCFStreamPropertySSLSettings,
|
||||
(CFTypeRef)settings);
|
||||
|
||||
inputStream = (bridge_stub NSInputStream *)readStream;
|
||||
[inputStream retain_stub];
|
||||
[inputStream setDelegate:self];
|
||||
[inputStream scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[inputStream open];
|
||||
|
||||
outputStream = (bridge_stub NSOutputStream *)writeStream;
|
||||
[outputStream retain_stub];
|
||||
[outputStream setDelegate:self];
|
||||
[outputStream scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[outputStream open];
|
||||
|
||||
|
||||
CFRelease(readStream);
|
||||
CFRelease(writeStream);
|
||||
}
|
||||
|
||||
|
||||
|
||||
self = [super initWithInputStream: inputStream outputStream: outputStream];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
#pragma mark -
|
||||
#pragma mark NSStreamDelegate
|
||||
- (void)stream:(NSStream *)aStream
|
||||
handleEvent:(NSStreamEvent)eventCode {
|
||||
switch (eventCode) {
|
||||
case NSStreamEventNone:
|
||||
break;
|
||||
case NSStreamEventHasBytesAvailable:
|
||||
break;
|
||||
case NSStreamEventOpenCompleted:
|
||||
break;
|
||||
case NSStreamEventHasSpaceAvailable:
|
||||
{
|
||||
SecPolicyRef policy = SecPolicyCreateSSL(NO, (__bridge CFStringRef)(sslHostname));
|
||||
SecTrustRef trust = NULL;
|
||||
CFArrayRef streamCertificatesRef =
|
||||
CFBridgingRetain((__bridge id)((__bridge CFArrayRef)([aStream propertyForKey:(NSString *) kCFStreamPropertySSLPeerCertificates])));
|
||||
SecTrustCreateWithCertificates(CFBridgingRetain((__bridge id)(streamCertificatesRef)),
|
||||
policy,
|
||||
&trust);
|
||||
|
||||
SecTrustResultType trustResultType = kSecTrustResultInvalid;
|
||||
SecTrustEvaluate(trust, &trustResultType);
|
||||
|
||||
BOOL proceed = NO;
|
||||
switch (trustResultType) {
|
||||
case kSecTrustResultProceed:
|
||||
proceed = YES;
|
||||
break;
|
||||
case kSecTrustResultUnspecified:
|
||||
NSLog(@"Trusted by OS");
|
||||
proceed = YES;
|
||||
break;
|
||||
case kSecTrustResultRecoverableTrustFailure:
|
||||
proceed = recoverFromTrustFailure(trust);
|
||||
break;
|
||||
case kSecTrustResultDeny:
|
||||
NSLog(@"Deny");
|
||||
break;
|
||||
case kSecTrustResultFatalTrustFailure:
|
||||
NSLog(@"FatalTrustFailure");
|
||||
break;
|
||||
case kSecTrustResultOtherError:
|
||||
NSLog(@"OtherError");
|
||||
break;
|
||||
case kSecTrustResultInvalid:
|
||||
NSLog(@"Invalid");
|
||||
break;
|
||||
default:
|
||||
NSLog(@"Default");
|
||||
break;
|
||||
}
|
||||
|
||||
if (trust) {
|
||||
CFRelease(trust);
|
||||
}
|
||||
if (policy) {
|
||||
CFRelease(policy);
|
||||
}
|
||||
if (!proceed) {
|
||||
NSLog(@"Cannot trust certificate. TrustResultType: %u", trustResultType);
|
||||
[aStream close];
|
||||
@throw [TSSLSocketException exceptionWithReason: @"Cannot trust certificate"];
|
||||
}
|
||||
}
|
||||
break;
|
||||
case NSStreamEventErrorOccurred:
|
||||
{
|
||||
NSError *theError = [aStream streamError];
|
||||
NSLog(@"Error occurred opening stream: %@", theError);
|
||||
// @throw [TSSLSocketException exceptionWithReason: @"Error occurred opening stream" error: theError];
|
||||
break;
|
||||
}
|
||||
case NSStreamEventEndEncountered:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
bool recoverFromTrustFailure(SecTrustRef myTrust)
|
||||
{
|
||||
|
||||
SecTrustResultType trustResult;
|
||||
OSStatus status = SecTrustEvaluate(myTrust, &trustResult);
|
||||
|
||||
CFAbsoluteTime trustTime,currentTime,timeIncrement,newTime;
|
||||
CFDateRef newDate;
|
||||
if (trustResult == kSecTrustResultRecoverableTrustFailure) {
|
||||
trustTime = SecTrustGetVerifyTime(myTrust);
|
||||
timeIncrement = 31536000;
|
||||
currentTime = CFAbsoluteTimeGetCurrent();
|
||||
newTime = currentTime - timeIncrement;
|
||||
if (trustTime - newTime){
|
||||
newDate = CFDateCreate(NULL, newTime);
|
||||
SecTrustSetVerifyDate(myTrust, newDate);
|
||||
status = SecTrustEvaluate(myTrust, &trustResult);
|
||||
}
|
||||
}
|
||||
if (trustResult != kSecTrustResultProceed) {
|
||||
NSLog(@"Certificate trust failure");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
- (void)close
|
||||
{
|
||||
if(self.mInput) {
|
||||
//Close and reset inputstream
|
||||
CFReadStreamSetProperty((__bridge CFReadStreamRef)(self.mInput), kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
[self.mInput setDelegate:nil];
|
||||
[self.mInput close];
|
||||
[self.mInput removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
self.mInput = nil;
|
||||
}
|
||||
|
||||
if(self.mOutput) {
|
||||
//Close and reset outputstream
|
||||
CFReadStreamSetProperty((__bridge CFReadStreamRef)(self.mOutput), kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
[self.mOutput setDelegate:nil];
|
||||
[self.mOutput close];
|
||||
[self.mOutput removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
self.mOutput = nil;
|
||||
}
|
||||
}
|
||||
|
||||
- (BOOL) isOpen
|
||||
{
|
||||
if(sd > 0)
|
||||
return TRUE;
|
||||
else
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
@end
|
||||
|
@ -1,42 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TSSLSocketException.h"
|
||||
|
||||
@implementation TSSLSocketException
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
error: (NSError *) error
|
||||
{
|
||||
NSDictionary * userInfo = nil;
|
||||
if (error != nil) {
|
||||
userInfo = [NSDictionary dictionaryWithObject: error forKey: @"error"];
|
||||
}
|
||||
|
||||
return [super exceptionWithName: @"TSSLSocketException"
|
||||
reason: reason
|
||||
userInfo: userInfo];
|
||||
}
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
{
|
||||
return [self exceptionWithReason: reason error: nil];
|
||||
}
|
||||
|
||||
@end
|
37
lib/cocoa/src/transport/TSSLSocketTransport.h
Normal file
37
lib/cocoa/src/transport/TSSLSocketTransport.h
Normal file
@ -0,0 +1,37 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TNSStreamTransport.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@interface TSSLSocketTransport : TNSStreamTransport <NSStreamDelegate>
|
||||
|
||||
-(id) initWithHostname:(NSString *)hostname
|
||||
port:(int)port
|
||||
error:(NSError **)error;
|
||||
|
||||
-(BOOL) isOpen;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
304
lib/cocoa/src/transport/TSSLSocketTransport.m
Normal file
304
lib/cocoa/src/transport/TSSLSocketTransport.m
Normal file
@ -0,0 +1,304 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
#import <Foundation/Foundation.h>
|
||||
#import <CoreFoundation/CoreFoundation.h>
|
||||
#import "TSSLSocketTransport.h"
|
||||
#import "TSSLSocketTransportError.h"
|
||||
#include <sys/socket.h>
|
||||
#include <netinet/in.h>
|
||||
#include <netdb.h>
|
||||
|
||||
#if !TARGET_OS_IPHONE
|
||||
#import <CoreServices/CoreServices.h>
|
||||
#else
|
||||
#import <CFNetwork/CFNetwork.h>
|
||||
#endif
|
||||
|
||||
@interface TSSLSocketTransport ()
|
||||
|
||||
@property(strong, nonatomic) NSString *sslHostname;
|
||||
@property(assign, nonatomic) int sd;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@implementation TSSLSocketTransport
|
||||
|
||||
-(id) initWithHostname:(NSString *)hostname
|
||||
port:(int)port
|
||||
error:(NSError **)error
|
||||
{
|
||||
_sslHostname = hostname;
|
||||
CFReadStreamRef readStream = NULL;
|
||||
CFWriteStreamRef writeStream = NULL;
|
||||
|
||||
|
||||
/* create a socket structure */
|
||||
struct sockaddr_in pin;
|
||||
struct hostent *hp = NULL;
|
||||
for (int i = 0; i < 10; i++) {
|
||||
|
||||
|
||||
|
||||
if ((hp = gethostbyname([hostname UTF8String])) == NULL) {
|
||||
NSLog(@"failed to resolve hostname %@", hostname);
|
||||
herror("resolv");
|
||||
if (i == 9) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TSSLSocketTransportErrorDomain
|
||||
code:TSSLSocketTransportErrorHostanameResolution
|
||||
userInfo:nil];
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
[NSThread sleepForTimeInterval:0.2];
|
||||
}
|
||||
else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
memset(&pin, 0, sizeof(pin));
|
||||
pin.sin_family = AF_INET;
|
||||
memcpy(&pin.sin_addr, hp->h_addr, sizeof(struct in_addr));
|
||||
pin.sin_port = htons(port);
|
||||
|
||||
/* create the socket */
|
||||
if ((_sd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1) {
|
||||
NSLog(@"failed to create socket for host %@:%d", hostname, port);
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TSSLSocketTransportErrorDomain
|
||||
code:TSSLSocketTransportErrorSocketCreate
|
||||
userInfo:nil];
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
|
||||
/* open a connection */
|
||||
if (connect(_sd, (struct sockaddr *)&pin, sizeof(pin)) == -1) {
|
||||
NSLog(@"failed to create conenct to host %@:%d", hostname, port);
|
||||
if (error) {
|
||||
*error = [NSError errorWithDomain:TSSLSocketTransportErrorDomain
|
||||
code:TSSLSocketTransportErrorConnect
|
||||
userInfo:nil];
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
CFStreamCreatePairWithSocket(kCFAllocatorDefault, _sd, &readStream, &writeStream);
|
||||
|
||||
CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
|
||||
NSInputStream *inputStream;
|
||||
NSOutputStream *outputStream;
|
||||
|
||||
if (readStream && writeStream) {
|
||||
|
||||
CFReadStreamSetProperty(readStream,
|
||||
kCFStreamPropertySocketSecurityLevel,
|
||||
kCFStreamSocketSecurityLevelTLSv1);
|
||||
|
||||
NSDictionary *settings = @{(__bridge NSString *)kCFStreamSSLValidatesCertificateChain: @YES};
|
||||
|
||||
CFReadStreamSetProperty((CFReadStreamRef)readStream,
|
||||
kCFStreamPropertySSLSettings,
|
||||
(CFTypeRef)settings);
|
||||
|
||||
CFWriteStreamSetProperty((CFWriteStreamRef)writeStream,
|
||||
kCFStreamPropertySSLSettings,
|
||||
(CFTypeRef)settings);
|
||||
|
||||
inputStream = (__bridge NSInputStream *)readStream;
|
||||
[inputStream setDelegate:self];
|
||||
[inputStream scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[inputStream open];
|
||||
|
||||
outputStream = (__bridge NSOutputStream *)writeStream;
|
||||
[outputStream setDelegate:self];
|
||||
[outputStream scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[outputStream open];
|
||||
|
||||
CFRelease(readStream);
|
||||
CFRelease(writeStream);
|
||||
}
|
||||
|
||||
self = [super initWithInputStream:inputStream outputStream:outputStream];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
-(void) dealloc
|
||||
{
|
||||
[self close];
|
||||
}
|
||||
|
||||
#pragma mark -
|
||||
#pragma mark NSStreamDelegate
|
||||
|
||||
-(void) stream:(NSStream *)aStream
|
||||
handleEvent:(NSStreamEvent)eventCode
|
||||
{
|
||||
switch (eventCode) {
|
||||
case NSStreamEventNone:
|
||||
break;
|
||||
|
||||
case NSStreamEventHasBytesAvailable:
|
||||
break;
|
||||
|
||||
case NSStreamEventOpenCompleted:
|
||||
break;
|
||||
|
||||
case NSStreamEventHasSpaceAvailable: {
|
||||
|
||||
BOOL proceed = NO;
|
||||
SecTrustResultType trustResult = kSecTrustResultInvalid;
|
||||
CFMutableArrayRef newPolicies = NULL;
|
||||
|
||||
do {
|
||||
|
||||
SecTrustRef trust = (__bridge SecTrustRef)[aStream propertyForKey:(NSString *)kCFStreamPropertySSLPeerTrust];
|
||||
|
||||
// Add new policy to current list of policies
|
||||
SecPolicyRef policy = SecPolicyCreateSSL(NO, (__bridge CFStringRef)(_sslHostname));
|
||||
if (!policy) {
|
||||
break;
|
||||
}
|
||||
|
||||
CFArrayRef policies;
|
||||
if (SecTrustCopyPolicies(trust, &policies) != errSecSuccess) {
|
||||
CFRelease(policy);
|
||||
break;
|
||||
}
|
||||
|
||||
newPolicies = CFArrayCreateMutableCopy(NULL, 0, policies);
|
||||
CFArrayAppendValue(newPolicies, policy);
|
||||
|
||||
CFRelease(policies);
|
||||
CFRelease(policy);
|
||||
|
||||
// Update trust policies
|
||||
if (SecTrustSetPolicies(trust, newPolicies) != errSecSuccess) {
|
||||
break;
|
||||
}
|
||||
|
||||
// Evaluate the trust chain
|
||||
if (SecTrustEvaluate(trust, &trustResult) != errSecSuccess) {
|
||||
break;
|
||||
}
|
||||
|
||||
switch (trustResult) {
|
||||
case kSecTrustResultProceed:
|
||||
// NSLog(@"Trusted by USER");
|
||||
proceed = YES;
|
||||
break;
|
||||
|
||||
case kSecTrustResultUnspecified:
|
||||
// NSLog(@"Trusted by OS");
|
||||
proceed = YES;
|
||||
break;
|
||||
|
||||
case kSecTrustResultRecoverableTrustFailure:
|
||||
proceed = recoverFromTrustFailure(trust, trustResult);
|
||||
break;
|
||||
|
||||
case kSecTrustResultDeny:
|
||||
// NSLog(@"Deny");
|
||||
break;
|
||||
|
||||
case kSecTrustResultFatalTrustFailure:
|
||||
// NSLog(@"FatalTrustFailure");
|
||||
break;
|
||||
|
||||
case kSecTrustResultOtherError:
|
||||
// NSLog(@"OtherError");
|
||||
break;
|
||||
|
||||
case kSecTrustResultInvalid:
|
||||
// NSLog(@"Invalid");
|
||||
break;
|
||||
|
||||
default:
|
||||
// NSLog(@"Default");
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
while (NO);
|
||||
|
||||
if (!proceed) {
|
||||
NSLog(@"TSSLSocketTransport: Cannot trust certificate. Result: %u", trustResult);
|
||||
[aStream close];
|
||||
}
|
||||
|
||||
if (newPolicies) {
|
||||
CFRelease(newPolicies);
|
||||
}
|
||||
|
||||
}
|
||||
break;
|
||||
|
||||
case NSStreamEventErrorOccurred: {
|
||||
NSLog(@"TSSLSocketTransport: Error occurred opening stream: %@", [aStream streamError]);
|
||||
break;
|
||||
}
|
||||
|
||||
case NSStreamEventEndEncountered:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
BOOL recoverFromTrustFailure(SecTrustRef myTrust, SecTrustResultType lastTrustResult)
|
||||
{
|
||||
CFAbsoluteTime trustTime = SecTrustGetVerifyTime(myTrust);
|
||||
CFAbsoluteTime currentTime = CFAbsoluteTimeGetCurrent();
|
||||
|
||||
CFAbsoluteTime timeIncrement = 31536000;
|
||||
CFAbsoluteTime newTime = currentTime - timeIncrement;
|
||||
|
||||
if (trustTime - newTime) {
|
||||
|
||||
CFDateRef newDate = CFDateCreate(NULL, newTime);
|
||||
SecTrustSetVerifyDate(myTrust, newDate);
|
||||
CFRelease(newDate);
|
||||
|
||||
if (SecTrustEvaluate(myTrust, &lastTrustResult) != errSecSuccess) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (lastTrustResult == kSecTrustResultProceed || lastTrustResult == kSecTrustResultUnspecified) {
|
||||
return YES;
|
||||
}
|
||||
|
||||
NSLog(@"TSSLSocketTransport: Unable to recover certificate trust failure");
|
||||
return YES;
|
||||
}
|
||||
|
||||
-(BOOL) isOpen
|
||||
{
|
||||
if (_sd > 0) {
|
||||
return TRUE;
|
||||
}
|
||||
else {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
30
lib/cocoa/src/transport/TSSLSocketTransportError.h
Normal file
30
lib/cocoa/src/transport/TSSLSocketTransportError.h
Normal file
@ -0,0 +1,30 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TTransportError.h"
|
||||
|
||||
|
||||
extern NSString *TSSLSocketTransportErrorDomain;
|
||||
|
||||
|
||||
typedef NS_ENUM (int, TSSLSocketTransportError) {
|
||||
TSSLSocketTransportErrorHostanameResolution = -10000,
|
||||
TSSLSocketTransportErrorSocketCreate = -10001,
|
||||
TSSLSocketTransportErrorConnect = -10002,
|
||||
};
|
23
lib/cocoa/src/transport/TSSLSocketTransportError.m
Normal file
23
lib/cocoa/src/transport/TSSLSocketTransportError.m
Normal file
@ -0,0 +1,23 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TSSLSocketTransportError.h"
|
||||
|
||||
|
||||
NSString *TSSLSocketTransportErrorDomain = @"TSSLSocketTransportErrorDomain";
|
@ -1,87 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
#import "TSocketClient.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
#if !TARGET_OS_IPHONE
|
||||
#import <CoreServices/CoreServices.h>
|
||||
#else
|
||||
#import <CFNetwork/CFNetwork.h>
|
||||
#endif
|
||||
|
||||
@interface TSocketClient ()
|
||||
{
|
||||
NSInputStream * inputStream;
|
||||
NSOutputStream * outputStream;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TSocketClient
|
||||
|
||||
- (id) initWithHostname: (NSString *) hostname
|
||||
port: (UInt32) port
|
||||
{
|
||||
inputStream = NULL;
|
||||
outputStream = NULL;
|
||||
CFReadStreamRef readStream = NULL;
|
||||
CFWriteStreamRef writeStream = NULL;
|
||||
CFStreamCreatePairWithSocketToHost(kCFAllocatorDefault, (bridge_stub CFStringRef)hostname, port, &readStream, &writeStream);
|
||||
if (readStream && writeStream) {
|
||||
CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
|
||||
inputStream = (bridge_stub NSInputStream *)readStream;
|
||||
[inputStream retain_stub];
|
||||
[inputStream setDelegate:self];
|
||||
[inputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[inputStream open];
|
||||
|
||||
outputStream = (bridge_stub NSOutputStream *)writeStream;
|
||||
[outputStream retain_stub];
|
||||
[outputStream setDelegate:self];
|
||||
[outputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[outputStream open];
|
||||
CFRelease(readStream);
|
||||
CFRelease(writeStream);
|
||||
}
|
||||
|
||||
self = [super initWithInputStream: inputStream outputStream: outputStream];
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
-(void)dealloc
|
||||
{
|
||||
[inputStream close];
|
||||
[inputStream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[inputStream setDelegate:nil];
|
||||
[inputStream release_stub];
|
||||
|
||||
[outputStream close];
|
||||
[outputStream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
||||
[outputStream setDelegate:nil];
|
||||
[outputStream release_stub];
|
||||
[super dealloc_stub];
|
||||
}
|
||||
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
@ -20,17 +20,15 @@
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TNSStreamTransport.h"
|
||||
|
||||
@interface TSocketClient : TNSStreamTransport
|
||||
#if TARGET_OS_IPHONE || (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_6)
|
||||
<NSStreamDelegate>
|
||||
#endif
|
||||
{
|
||||
}
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
- (id) initWithHostname: (NSString *) hostname
|
||||
port: (UInt32) port;
|
||||
|
||||
@interface TSocketTransport : TNSStreamTransport
|
||||
|
||||
-(id) initWithHostname:(NSString *)hostname
|
||||
port:(int)port;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
73
lib/cocoa/src/transport/TSocketTransport.m
Normal file
73
lib/cocoa/src/transport/TSocketTransport.m
Normal file
@ -0,0 +1,73 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
#import "TSocketTransport.h"
|
||||
|
||||
#if !TARGET_OS_IPHONE
|
||||
#import <CoreServices/CoreServices.h>
|
||||
#else
|
||||
#import <CFNetwork/CFNetwork.h>
|
||||
#endif
|
||||
|
||||
@interface TSocketTransport () <NSStreamDelegate>
|
||||
@end
|
||||
|
||||
|
||||
@implementation TSocketTransport
|
||||
|
||||
-(id) initWithHostname:(NSString *)hostname
|
||||
port:(int)port
|
||||
{
|
||||
NSInputStream *inputStream = nil;
|
||||
NSOutputStream *outputStream = nil;
|
||||
|
||||
CFReadStreamRef readStream = NULL;
|
||||
CFWriteStreamRef writeStream = NULL;
|
||||
CFStreamCreatePairWithSocketToHost(kCFAllocatorDefault, (__bridge CFStringRef)hostname, port, &readStream, &writeStream);
|
||||
if (readStream && writeStream) {
|
||||
|
||||
CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
|
||||
|
||||
inputStream = (__bridge NSInputStream *)readStream;
|
||||
[inputStream setDelegate:self];
|
||||
[inputStream scheduleInRunLoop:NSRunLoop.mainRunLoop forMode:NSDefaultRunLoopMode];
|
||||
[inputStream open];
|
||||
|
||||
outputStream = (__bridge NSOutputStream *)writeStream;
|
||||
[outputStream setDelegate:self];
|
||||
[outputStream scheduleInRunLoop:NSRunLoop.mainRunLoop forMode:NSDefaultRunLoopMode];
|
||||
[outputStream open];
|
||||
}
|
||||
else {
|
||||
|
||||
if (readStream) {
|
||||
CFRelease(readStream);
|
||||
}
|
||||
|
||||
if (writeStream) {
|
||||
CFRelease(writeStream);
|
||||
}
|
||||
|
||||
return nil;
|
||||
}
|
||||
|
||||
return [super initWithInputStream:inputStream outputStream:outputStream];
|
||||
}
|
||||
|
||||
@end
|
@ -17,20 +17,32 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
||||
@protocol TTransport <NSObject>
|
||||
|
||||
/**
|
||||
* Guarantees that all of len bytes are read
|
||||
*
|
||||
* @param buf Buffer to read into
|
||||
* @param offset Index in buffer to start storing bytes at
|
||||
* @param length Maximum number of bytes to read
|
||||
* @return The number of bytes actually read, which must be equal to len
|
||||
* @throws TTransportException if there was an error reading data
|
||||
*/
|
||||
- (size_t) readAll: (uint8_t *) buf offset: (size_t) offset length: (size_t) length;
|
||||
/**
|
||||
* Guarantees that all of len bytes are read
|
||||
*
|
||||
* @param buf Buffer to read into
|
||||
* @param off Index in buffer to start storing bytes at
|
||||
* @param len Maximum number of bytes to read
|
||||
* @return YES if succeeded, NO if failed
|
||||
* @throws TTransportError if there was an error reading data
|
||||
*/
|
||||
-(BOOL) readAll:(UInt8 *)buf offset:(UInt32)off length:(UInt32)len error:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) write: (const uint8_t *) data offset: (size_t) offset length: (size_t) length;
|
||||
-(UInt32) readAvail:(UInt8 *)buf offset:(UInt32)off maxLength:(UInt32)maxLen error:(NSError *__autoreleasing *)error;
|
||||
|
||||
-(BOOL) write:(const UInt8 *)data offset:(UInt32)offset length:(UInt32)length error:(NSError *__autoreleasing *)error;
|
||||
|
||||
-(BOOL) flush:(NSError *__autoreleasing *)error;
|
||||
|
||||
- (void) flush;
|
||||
@end
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
@ -17,26 +17,27 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import "TTransport.h"
|
||||
#import "TError.h"
|
||||
|
||||
@interface THTTPClient : NSObject <TTransport> {
|
||||
NSURL * mURL;
|
||||
NSMutableURLRequest * mRequest;
|
||||
NSMutableData * mRequestData;
|
||||
NSData * mResponseData;
|
||||
size_t mResponseDataOffset;
|
||||
NSString * mUserAgent;
|
||||
int mTimeout;
|
||||
}
|
||||
|
||||
- (id) initWithURL: (NSURL *) aURL;
|
||||
extern NSString *TTransportErrorDomain;
|
||||
|
||||
- (id) initWithURL: (NSURL *) aURL
|
||||
userAgent: (NSString *) userAgent
|
||||
timeout: (int) timeout;
|
||||
|
||||
- (void) setURL: (NSURL *) aURL;
|
||||
typedef NS_ENUM (int, TTransportError) {
|
||||
TTransportErrorUnknown = 0,
|
||||
TTransportErrorNotOpen = 1,
|
||||
TTransportErrorAlreadyOpen = 2,
|
||||
TTransportErrorTimedOut = 3,
|
||||
TTransportErrorEndOfFile = 4,
|
||||
};
|
||||
|
||||
@end
|
||||
|
||||
extern NSString *TTransportErrorExtendedErrorKey;
|
||||
extern NSString *TTransportErrorHttpErrorKey;
|
||||
|
||||
|
||||
typedef NS_ENUM(int, THttpTransportError) {
|
||||
THttpTransportErrorInvalidResponse = 1001,
|
||||
THttpTransportErrorInvalidStatus = 1002,
|
||||
THttpTransportErrorAuthentication = 1003,
|
||||
};
|
@ -17,13 +17,11 @@
|
||||
* under the License.
|
||||
*/
|
||||
|
||||
#import "TTransportException.h"
|
||||
#import "TTransportError.h"
|
||||
|
||||
@interface TSSLSocketException : TTransportException
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
error: (NSError *) error;
|
||||
NSString *TTransportErrorDomain = @"TTransportErrorDomain";
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason;
|
||||
|
||||
@end
|
||||
NSString *TTransportErrorExtendedErrorKey = @"extendedError";
|
||||
NSString *TTransportErrorHttpErrorKey = @"httpError";
|
@ -1,44 +0,0 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF licenses this file
|
||||
* to you 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.
|
||||
*/
|
||||
|
||||
#import "TTransportException.h"
|
||||
#import "TObjective-C.h"
|
||||
|
||||
@implementation TTransportException
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
error: (NSError *) error
|
||||
{
|
||||
NSDictionary * userInfo = nil;
|
||||
if (error != nil) {
|
||||
userInfo = [NSDictionary dictionaryWithObject: error forKey: @"error"];
|
||||
}
|
||||
|
||||
return [super exceptionWithName: @"TTransportException"
|
||||
reason: reason
|
||||
userInfo: userInfo];
|
||||
}
|
||||
|
||||
|
||||
+ (id) exceptionWithReason: (NSString *) reason
|
||||
{
|
||||
return [self exceptionWithReason: reason error: nil];
|
||||
}
|
||||
|
||||
@end
|
Loading…
Reference in New Issue
Block a user