File: DKIntrospectionParserDelegate.m

package info (click to toggle)
dbuskit 0.1.1-14
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 3,956 kB
  • sloc: objc: 10,543; sh: 9,463; ansic: 200; makefile: 32
file content (261 lines) | stat: -rw-r--r-- 7,304 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
/** Implementation of the DKIntrospectionParserDelegate helper class.
   Copyright (C) 2010 Free Software Foundation, Inc.

   Written by:  Niels Grewe <niels.grewe@halbordnung.de>
   Created: July 2010

   This library is free software; you can redistribute it and/or
   modify it under the terms of the GNU Lesser General Public
   License as published by the Free Software Foundation; either
   version 2 of the License, or (at your option) any later version.

   This library is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
   Library General Public License for more details.

   You should have received a copy of the GNU Lesser General Public
   License along with this library; if not, write to the Free
   Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
   Boston, MA 02111 USA.
   */


#import "DKIntrospectionParserDelegate.h"

#import "DKArgument.h"
#import "DKInterface.h"
#import "DKIntrospectionNode.h"
#import "DKMethod.h"
#import "DKObjectPathNode.h"
#import "DKProperty.h"
#import "DKSignal.h"

#import <Foundation/NSArray.h>
#import <Foundation/NSDebug.h>
#import <Foundation/NSDictionary.h>
#import <Foundation/NSException.h>
#import <Foundation/NSNull.h>
#import <Foundation/NSXMLParser.h>

@interface DKIntrospectionParserDelegate (StackManagement)
- (void)pushToStack: (id)obj;
- (void)popStack;
- (id)leaf;
@end

@implementation DKIntrospectionParserDelegate

- (id) initWithParentForNodes: (id)parent
{
  if (nil == (self = [super init]))
  {
    return nil;
  }

  stack = [[NSMutableArray alloc] init];
  [self pushToStack: parent];
  return self;
}

- (void) dealloc
{
  [stack release];
  [super dealloc];
}

- (id)leaf
{
  id object = [stack objectAtIndex: ([stack count] - 1)];
  if ([[NSNull null] isEqual: object])
  {
    return nil;
  }
  return object;
}

- (void)popStack
{
  NSUInteger count = [stack count];
  if (0 != count)
  {
    [stack removeObjectAtIndex: (count - 1) ];
  }
}

- (void)pushToStack: (id)obj
{
  if (nil == obj)
  {
    obj = [NSNull null];
  }
  [stack addObject: obj];
}

/* Parser delegate methods */
- (void) parserDidStartDocument: (NSXMLParser*)aParser
{
  NSDebugMLog(@"Started parsing XML");
}

- (void) parserDidEndDocument: (NSXMLParser*)aParser
{
  NSDebugMLog(@"Stopped parsing XML");
}

- (void) parser: (NSXMLParser*)aParser
didStartElement: (NSString*)aNode
   namespaceURI: (NSString*)aNamespaceURI
  qualifiedName: (NSString*)aQualifierName
     attributes: (NSDictionary*)someAttributes
{
  NSString *theName = [someAttributes objectForKey: @"name"];
  DKIntrospectionNode *newNode = nil;
  id leaf = [self leaf];
  BOOL isRoot = (0 == xmlDepth);
  // Ignore the !doctype if the parser exposes it.
  if (isRoot && [@"!doctype" isEqualToString: aNode])
  {
    return;
  }
  xmlDepth++;
  NSDebugLog(@"Starting <%@> node '%@' at depth %lu.",
    aNode,
    theName,
    xmlDepth);

  if ([@"node" isEqualToString: aNode])
  {
    if ([theName length] > 0)
    {
      if (isRoot && ('/' != [theName characterAtIndex: 0]))
      {
	// relative paths must refer to nodes contained in the main node.
	[NSException raise: @"DKIntrospectionException"
	            format: @"Introspection data contains invalid root node named '%@'",
	  theName];
      }
    }

    if (isRoot)
    {
      // For the root node, we just push the leaf we got initially once again:
      newNode = RETAIN(leaf);
    }
    else
    {
      newNode = [[DKObjectPathNode alloc] initWithName: theName
                                                parent: leaf];
      if ([leaf conformsToProtocol: @protocol(DKObjectPathNode)])
      {
        [(id<DKObjectPathNode>)leaf _addChildNode: (DKObjectPathNode*)newNode];
      }
    }
  }
  else if (([@"interface" isEqualToString: aNode]) && ([theName length] > 0))
  {
    newNode = [[DKInterface alloc] initWithName: theName
                                         parent: leaf];
      if ([leaf conformsToProtocol: @protocol(DKObjectPathNode)])
      {
	[(id<DKObjectPathNode>)leaf _addInterface: (DKInterface*)newNode];
      }
  }
  else if (([@"annotation" isEqualToString: aNode]) && ([theName length] > 0))
  {
    id theValue = [someAttributes objectForKey: @"value"];
    if (nil == theValue)
    {
      theValue = [NSNull null];
    }
    if ([leaf respondsToSelector: @selector(setAnnotationValue:forKey:)])
    {
      [leaf setAnnotationValue: theValue
                        forKey: theName];
    }
  }
  else if ([leaf isKindOfClass: [DKInterface class]])
  {
    // Things that should only appear in interfaces (methods, signals,
    // porperties):
    DKInterface *ifLeaf = (DKInterface*)leaf;
    if ([@"method" isEqualToString: aNode])
    {
      newNode = [[DKMethod alloc] initWithName: theName
                                        parent: leaf];
      [ifLeaf addMethod: (DKMethod*)newNode];
    }
    else if ([@"signal" isEqualToString: aNode])
    {
      newNode = [[DKSignal alloc] initWithName: theName
                                        parent: leaf];
      [ifLeaf addSignal: (DKSignal*)newNode];
    }
    else if ([@"property" isEqualToString: aNode])
    {
      const char *type = [[someAttributes objectForKey: @"type"] UTF8String];
      NSString *access = [someAttributes objectForKey: @"access"];
      newNode = [[DKProperty alloc] initWithDBusSignature: type
                                         accessAttributes: access
                                                     name: theName
                                                   parent: leaf];
      [ifLeaf addProperty: (DKProperty*)newNode];
    }
  }
  else if (([leaf isKindOfClass: [DKMethod class]])
    || [leaf isKindOfClass: [DKSignal class]])
  {
    // Arguments should only appear in methods or signals
    if ([@"arg" isEqualToString: aNode])
    {
      NSString *direction = [someAttributes objectForKey: @"direction"];
      const char *type = [[someAttributes objectForKey: @"type"] UTF8String];
      newNode = [[DKArgument alloc] initWithDBusSignature: type
                                                     name: theName
						   parent: leaf];
      // DKSignal also implements -addArgument:direction: with the same
      // signature.
      [(DKMethod*)leaf addArgument: (DKArgument*)newNode
                         direction: direction];
    }
  }
  else
  {
    NSDebugMLog(@"Ignoring <%@> node '%@' at depth %lu.",
      aNode,
      theName,
      xmlDepth);
    newNode = [[DKIntrospectionNode alloc] initWithName: theName
                                                 parent: leaf];
  }

  [self pushToStack: newNode];

  if (newNode != nil)
  {
    // We did not autorelease the nodes when creating them, so we release them
    // here:
    [newNode release];
  }
}

- (void) parser: (NSXMLParser*)aParser
  didEndElement: (NSString*)aNode
   namespaceURI: (NSString*)aNamespaceURI
  qualifiedName: (NSString*)aQualifierName
{
  // Ignore the doctype if the parser exposes it
  if ((0 == xmlDepth) && [@"!doctype" isEqualToString: aNode])
  {
    return;
  }
  NSDebugMLog(@"Ended node: %@", aNode);
  xmlDepth--;
  [self popStack];
  if (0 == xmlDepth)
  {
    NSDebugMLog(@"Ended parsing");
  }
}

@end