View Javadoc

1   /*
2    $Id: Node.java,v 1.10 2005/08/10 09:53:58 hmeling Exp $
3   
4    Copyright 2003 (C) James Strachan and Bob Mcwhirter. All Rights Reserved.
5   
6    Redistribution and use of this software and associated documentation
7    ("Software"), with or without modification, are permitted provided
8    that the following conditions are met:
9   
10   1. Redistributions of source code must retain copyright
11      statements and notices.  Redistributions must also contain a
12      copy of this document.
13  
14   2. Redistributions in binary form must reproduce the
15      above copyright notice, this list of conditions and the
16      following disclaimer in the documentation and/or other
17      materials provided with the distribution.
18  
19   3. The name "groovy" must not be used to endorse or promote
20      products derived from this Software without prior written
21      permission of The Codehaus.  For written permission,
22      please contact info@codehaus.org.
23  
24   4. Products derived from this Software may not be called "groovy"
25      nor may "groovy" appear in their names without prior written
26      permission of The Codehaus. "groovy" is a registered
27      trademark of The Codehaus.
28  
29   5. Due credit should be given to The Codehaus -
30      http://groovy.codehaus.org/
31  
32   THIS SOFTWARE IS PROVIDED BY THE CODEHAUS AND CONTRIBUTORS
33   ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT
34   NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
35   FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
36   THE CODEHAUS OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
37   INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
38   (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
39   SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
40   HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
41   STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
42   ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
43   OF THE POSSIBILITY OF SUCH DAMAGE.
44  
45   */
46  package groovy.util;
47  
48  import java.io.PrintWriter;
49  import java.util.ArrayList;
50  import java.util.Collection;
51  import java.util.Collections;
52  import java.util.Iterator;
53  import java.util.List;
54  import java.util.Map;
55  
56  import org.codehaus.groovy.runtime.InvokerHelper;
57  
58  /***
59   * Represents an arbitrary tree node which can be used for structured  metadata which can be any arbitrary XML-like tree.
60   * A node can have a name, a value and an optional Map of attributes.
61   * Typically the name is a String and a value is either a String or a List of other Nodes.
62   * Though the types are extensible to provide a flexible structure. 
63   * e.g. you could use a QName as the name which includes a namespace URI and a local name. Or a JMX ObjectName etc.
64   * So this class can represent metadata like {foo a=1 b="abc"} or nested metadata like {foo a=1 b="123" { bar x=12 text="hello" }}
65   * 
66   * @author <a href="mailto:james@coredevelopers.net">James Strachan</a>
67   * @version $Revision: 1.10 $
68   */
69  public class Node implements java.io.Serializable {
70  
71      private Node parent;
72      private Object name;
73      private Map attributes;
74      private Object value;
75  
76      public Node(Node parent, Object name) {
77          this(parent, name, Collections.EMPTY_MAP, Collections.EMPTY_LIST);
78      }
79  
80      public Node(Node parent, Object name, Object value) {
81          this(parent, name, Collections.EMPTY_MAP, value);
82      }
83  
84      public Node(Node parent, Object name, Map attributes) {
85          this(parent, name, attributes, Collections.EMPTY_LIST);
86      }
87  
88      public Node(Node parent, Object name, Map attributes, Object value) {
89          this.parent = parent;
90          this.name = name;
91          this.attributes = attributes;
92          this.value = value;
93          
94          if (parent != null) {
95              Object parentValue = parent.value();
96              List parentList = null;
97              if (parentValue instanceof List) {
98                  parentList = (List) parentValue;
99              }
100             else {
101                 parentList = new ArrayList();
102                 parentList.add(parentValue);
103                 parent.setValue(parentList);
104             }
105             parentList.add(this);
106         }
107     }
108 
109     public String text() {
110         if (value instanceof String) {
111             return (String) value;
112         }
113         else if (value instanceof Collection) {
114             Collection coll = (Collection) value;
115             String previousText = null;
116             StringBuffer buffer = null;
117             for (Iterator iter = coll.iterator(); iter.hasNext();) {
118                 Object child = iter.next();
119                 if (child instanceof String) {
120                     String childText = (String) child;
121                     if (previousText == null) {
122                         previousText = childText;
123                     }
124                     else {
125                         if (buffer == null) {
126                             buffer = new StringBuffer();
127                             buffer.append(previousText);
128                         }
129                         buffer.append(childText);
130                     }
131                 }
132             }
133             if (buffer != null) {
134                 return buffer.toString();
135             }
136             else {
137                 if (previousText != null) {
138                     return previousText;
139                 }
140             }
141         }
142         return "";
143     }
144 
145     public Iterator iterator() {
146         return children().iterator();
147     }
148     
149     public List children() {
150         if (value == null) {
151             return Collections.EMPTY_LIST;
152         }
153         else if (value instanceof List) {
154             return (List) value;
155         }
156         else {
157             // we're probably just a String
158             return Collections.singletonList(value);
159         }
160     }
161 
162     public Map attributes() {
163         return attributes;
164     }
165 
166     public Object attribute(Object key) {
167         return (attributes != null) ? attributes.get(key) : null;
168     }
169     
170     public Object name() {
171         return name;
172     }
173 
174     public Object value() {
175         return value;
176     }
177 
178     public void setValue(Object value) {
179         this.value = value;
180     }
181 
182     public Node parent() {
183         return parent;
184     }
185 
186     public Object get(String key) {
187         if (key.charAt(0) == '@') {
188             String attributeName = key.substring(1);
189             return attributes().get(attributeName);
190         }
191         else {
192             // iterate through list looking for node with name 'key'
193             List answer = new ArrayList();
194             for (Iterator iter = children().iterator(); iter.hasNext();) {
195                 Object child = iter.next();
196                 if (child instanceof Node) {
197                     Node childNode = (Node) child;
198                     Object childNodeName = childNode.name();
199                     if (childNodeName != null && childNodeName.equals(key)) {
200                         answer.add(childNode);
201                     }
202                 }
203             }
204             return answer;
205         }
206     }
207 
208 //    public Object get(int idx) {
209 //        return children().get(idx);
210 //    }
211 
212 
213 
214     /***
215      * Provide a collection of all the nodes in the tree
216      * using a depth first traversal
217      */
218     public List depthFirst() {
219         List answer = new ArrayList();
220         answer.add(this);
221         answer.addAll(depthFirstRest());
222         return answer;
223     }
224     
225     private  List depthFirstRest() {
226         List answer = new ArrayList();
227         for (Iterator iter = InvokerHelper.asIterator(value); iter.hasNext(); ) {
228             Object child = iter.next();
229             if (child instanceof Node) {
230                 Node childNode = (Node) child;
231                 List children = childNode.depthFirstRest();
232                 answer.add(childNode);
233                 answer.addAll(children);
234             }
235         }
236         return answer;
237     }
238 
239     /***
240      * Provide a collection of all the nodes in the tree
241      * using a bredth first traversal
242      */
243     public List breadthFirst() {
244         List answer = new ArrayList();
245         answer.add(this);
246         answer.addAll(breadthFirstRest());
247         return answer;
248     }
249     
250     private  List breadthFirstRest() {
251         List answer = new ArrayList();
252         for (Iterator iter = InvokerHelper.asIterator(value); iter.hasNext(); ) {
253             Object child = iter.next();
254             if (child instanceof Node) {
255                 Node childNode = (Node) child;
256                 answer.add(childNode);
257             }
258         }
259         List copy = new ArrayList(answer);
260         for (Iterator iter = copy.iterator(); iter.hasNext(); ) {
261             Node childNode = (Node) iter.next();
262             List children = childNode.breadthFirstRest();
263             answer.addAll(children);
264         }
265         return answer;
266     }
267 
268     public String toString() {
269         return name + "[attributes=" + attributes + "; value=" + value + "]";
270     }
271 
272     public void print(PrintWriter out) {
273         new NodePrinter(out).print(this);
274     }
275 }