Once you obtain WSDL2ObjC, code generation is pretty simple.
- Launch the app
- Browse to a WSDL file or enter in a URL
- Browse to an output directory
- Click "Parse WSDL"
Source code files will be added to the output directory you've specified. There will be one pair of .h/.m files for each namespace in your WSDL.
You can add the output files to your project or create a web service framework from them. Each project that uses the generated web service code will need to link against libxml2 by performing the following for each target in your XCode project:
- Go to the General tab of the project target
- Add libxml2.dylib to Linked Frameworks and Libraries
- If building for iOS, also add CFNetworking.framework
- Go to the Build Settings tab
- Add $(SDKROOT)/usr/include/libxml2 to the Header Search Paths property
You can use a given web service as follows:
#import "MyWebService.h"
MyWebServiceBinding *binding = [MyWebService MyWebServiceBinding];
binding.logXMLInOut = YES;
ns1_MyOperationRequest *request = [ns1_MyOperationRequest new];
request.attribute = @"attributeValue";
request.element = [ns1_MyElement new];
request.element.value = @"elementValue"];
MyWebServiceBindingResponse *response = [binding myOperationUsingParameters:request];
NSArray *responseHeaders = response.headers;
NSArray *responseBodyParts = response.bodyParts;
for (id header in responseHeaders) {
if ([header isKindOfClass:[ns2_MyHeaderResponse class]]) {
ns2_MyHeaderResponse *headerResponse = (ns2_MyHeaderResponse*)header;
// ... Handle ns2_MyHeaderResponse ...
}
}
for (id bodyPart in responseBodyParts) {
if ([bodyPart isKindOfClass:[ns2_MyBodyResponse class]]) {
ns2_MyBodyResponse *body = (ns2_MyBodyResponse*)bodyPart;
// ... Handle ns2_MyBodyResponse ...
}
}
Assume the following:
- A SOAP service called "Friends"
- A SOAP method called GetFavoriteColor that has a request attribute called Friend, and a response attribute called Color (i.e. you're asking it to return you the favorite color for a given a friend)
- All the methods in this service ask for basic HTTP authentication, using a username and password that you acquired from the user via text fields
- (IBAction)pressedRequestButton:(id)sender {
FriendsBinding *bFriends = [FriendsService FriendsBinding];
bFriends.logXMLInOut = YES;
bFriends.authUsername = u.text;
bFriends.authPassword = p.text;
types_getFavoriteColorRequestType *cRequest = [types_getFavoriteColorRequestType new];
cRequest.friend = @"Johnny";
[bFriends getFavoriteColorAsyncUsingRequest:cRequest delegate:self];
}
- (void) operation:(FriendsBindingOperation *)operation completedWithResponse:(FriendsBindingResponse *)response {
NSArray *responseHeaders = response.headers;
NSArray *responseBodyParts = response.bodyParts;
for (id header in responseHeaders) {
// here do what you want with the headers, if there's anything of value in them
}
for (id bodyPart in responseBodyParts) {
/****
* SOAP Fault Error
****/
if ([bodyPart isKindOfClass:[SOAPFault class]]) {
// You can get the error like this:
tV.text = ((SOAPFault *)bodyPart).simpleFaultString;
continue;
}
/****
* Get Favorite Color
****/
if ([bodyPart isKindOfClass:[types_getFavoriteColorResponseType class]]) {
types_getFavoriteColorResponseType *body = (types_getFavoriteColorResponseType*)bodyPart;
// Now you can extract the color from the response
q.text = body.color;
continue;
}
// ...
}
}
If a WSDL has defined a string type that has attributes, then wsdl2objc will map it to a generic NSObject with a property called "content" which will hold the actual string. So if you want to use the string, you have to call object.content. If you want its attributes, they're also properties of the object. The short reason for this is that Cocoa makes it very hard to subclass NSStrings.
Assume the following:
- A SOAP service called "Tws"
- A SOAP method called Hello that has a request attribute called Name, and a response attribute called Return
//
// ViewController.m
// aaaa
//
// Created by Denis Chepil on 31.05.14.
// Copyright (c) 2014 Chepil. All rights reserved.
//
#import "ViewController.h"
#import "tws.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
tws_twsPortBinding *b = [tws twsPortBinding];
[b setLogXMLInOut:YES];
tws_hello *s = [[tws_hello alloc] init];
[s setV_name:@"myexample"];
//[b helloUsingHello:s];
[b helloUsingHello:s
success:^(NSArray *headers, NSArray *bodyParts) {
//headers
for (int i=0; i<[headers count]; i++) {
NSLog(@"i: %d, %@", i, [headers objectAtIndex:i]);
}
//bodyParts
for (int i=0; i<[bodyParts count]; i++) {
NSLog(@"res: %@",[(tws_helloResponse*)[bodyParts objectAtIndex:i] v_return]);
}
}
error:^(NSError *error) {
NSLog(@"error: %@", error.description);
}
];
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end