View Javadoc

1   /**
2    *    Copyright 2011 meltmedia
3    *
4    *    Licensed under the Apache License, Version 2.0 (the "License");
5    *    you may not use this file except in compliance with the License.
6    *    You may obtain a copy of the License at
7    *
8    *        http://www.apache.org/licenses/LICENSE-2.0
9    *
10   *    Unless required by applicable law or agreed to in writing, software
11   *    distributed under the License is distributed on an "AS IS" BASIS,
12   *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   *    See the License for the specific language governing permissions and
14   *    limitations under the License.
15   */
16  package org.xchain.namespaces.sax;
17  
18  import org.apache.commons.jxpath.JXPathContext;
19  
20  import org.xchain.Command;
21  import org.xchain.annotations.Attribute;
22  import org.xchain.annotations.AttributeType;
23  import org.xchain.annotations.Element;
24  import org.xchain.framework.net.UrlFactory;
25  
26  import java.net.URL;
27  
28  import java.io.OutputStream;
29  import java.io.File;
30  import java.io.FileOutputStream;
31  import java.io.Writer;
32  
33  import javax.xml.transform.dom.DOMResult;
34  import javax.xml.transform.sax.SAXResult;
35  import javax.xml.transform.stream.StreamResult;
36  import javax.xml.transform.Result;
37  
38  import org.xml.sax.ContentHandler;
39  
40  import org.w3c.dom.Node;
41  
42  import org.slf4j.Logger;
43  import org.slf4j.LoggerFactory;
44  
45  /**
46   * <p>The &lt;sax:result/&gt; command adds result objects to a pipeline.  This command can add several types of results, based on the attributes
47   * defined on the command.  To write output to a system id, add the system-id attribute to the command:</p>
48   *
49   * <code class="source">
50   * &lt;sax:pipeline xmlns:sax="http://www.xchain.org/sax/1.0"&gt;
51   *   ...
52   *   &lt;sax:result system-id="'file:/some/path'"&gt;
53   * &lt;/sax:pipeline&gt;
54   * </code>
55   *
56   * <p>To write output to a file path, add the path attribute:</p>
57   *
58   * <code class="source">
59   * &lt;sax:pipeline xmlns:sax="http://www.xchain.org/sax/1.0"&gt;
60   *   ...
61   *   &lt;sax:result path="'/some/path'"&gt;
62   * &lt;/sax:pipeline&gt;
63   * </code>
64   *
65   * <p>To write output to an object in the context, add the select attribute:</p>
66   *
67   * <code class="source">
68   * &lt;sax:pipeline xmlns:sax="http://www.xchain.org/sax/1.0"&gt;
69   *   ...
70   *   &lt;sax:result select="$result"&gt;
71   * &lt;/sax:pipeline&gt;
72   * </code>
73   *
74   * <p>The value of the select attribute must resovle to one of:</p>
75   * <ul>
76   *   <li>javax.xml.transform.Result</li>
77   *   <li>org.xml.sax.ContentHandler</li>
78   *   <li>org.w3c.dom.Node</li>
79   *   <li>java.io.OutputStream</li>
80   *   <li>java.io.Writer</li>
81   *   <li>java.io.File</li>
82   * </ul>
83   *
84   * @author Mike Moulton
85   * @author Christian Trimble
86   * @author Devon Tackett
87   * @author Jason Rose
88   * @author Josh Kennedy
89   */
90  @Element(localName = "result")
91  public abstract class ResultCommand implements Command {
92    public static Logger log = LoggerFactory.getLogger(ResultCommand.class);
93  
94    /**
95     * <p>DEPRICATED: Use system-id instead.  The system id to send the output to.</p>
96     * @param context the JXPathContext to evaluate against.
97     */
98    @Attribute(localName = "systemId", type = AttributeType.JXPATH_VALUE)
99    public abstract String getSystemIdDepricated(JXPathContext context)
100     throws Exception;
101 
102   /**
103    * <p>Returns true if the systemId attribute has been set.</p>
104    * @return true if the systemId attribute has been set, false otherwise.
105    */
106   public abstract boolean hasSystemIdDepricated();
107 
108   /**
109    * <p>The system id to send output to.</p>
110    * @param context the JXPathContext to evaluate against.
111    */
112   @Attribute(localName="system-id", type = AttributeType.JXPATH_VALUE)
113   public abstract String getSystemId(JXPathContext context)
114     throws Exception;
115 
116   /**
117    * <p>Returns true if the system-id attribute has been set.</p>
118    * @return true if the system-id attribute has been set, false otherwise.
119    */
120   public abstract boolean hasSystemId();
121 
122   /**
123    * <p>The file path to send output to.</p>
124    * @param context the JXPathContext to evaluate against.
125    */
126   @Attribute(localName = "path", type = AttributeType.JXPATH_VALUE)
127   public abstract String getPath(JXPathContext context)
128     throws Exception;
129 
130   /**
131    * <p>Returns true if the path attribute has been set.</p>
132    * @return true if the path attribute has been set, false otherwise.
133    */
134   public abstract boolean hasPath();
135 
136   /**
137    * <p>The object to send output to.</p>
138    * @param context the JXPathContext to evaluate against.
139    */
140   @Attribute(localName = "select", type = AttributeType.JXPATH_SELECT_SINGLE_NODE)
141   public abstract Object getSelect(JXPathContext context)
142     throws Exception;
143 
144   /**
145    * <p>Returns true if the select attribute has been set.</p>
146    * @return true if the select attribute has been set, false otherwise.
147    */
148   public abstract boolean hasSelect();
149 
150   /**
151    * <p>Returns the result for the select attribute.</p>
152    * @param context the JXPathContext to evaluate against.
153    * @return the correct result object for the type of object selected from the context.
154    */
155   public Result createResultForSelect(JXPathContext context)
156     throws Exception
157   {
158     Object object = getSelect(context);
159 
160     if( object == null ) {
161       throw new IllegalArgumentException("The selected object cannot be null.");
162     }
163     // if the object is a result, then use it.
164     else if( object instanceof Result ) {
165       return (Result) object;
166     }
167 
168     // if the object is a stream, then create a stream source.
169     else if( object instanceof OutputStream ) {
170       return new StreamResult((OutputStream) object);
171     } else if( object instanceof Writer ) {
172       return new StreamResult((Writer) object);
173     } else if( object instanceof File ) {
174       return new StreamResult((File) object);
175     }
176 
177     // if the object is a content handler, then create a sax result.
178     else if( object instanceof ContentHandler ) {
179       return new SAXResult((ContentHandler) object);
180     }
181 
182     // if the result is a node, then create a dom result.
183     else if( object instanceof Node ) {
184       return new DOMResult((Node) object);
185     }
186 
187     // we do not how to make a result for this object, so bail out.
188     else {
189       throw new IllegalArgumentException("The selected result object (" + object.getClass().getName() + ") is not a result object nor is it an output stream.");
190     }
191   }
192 
193   /**
194    * <p>Returns the Result object for the system-id attribute.</p>
195    * @param context the JXPathContext to evaluate against.
196    * @return a stream result for the system id.
197    */
198   public Result createResultForSystemId(JXPathContext context)
199     throws Exception
200   {
201     // set the system id.
202     //String systemId = getSystemId(context);
203     String systemId = null;
204     if( hasSystemId() ) {
205       systemId = getSystemId(context);
206     }
207     else {
208       systemId = getSystemIdDepricated(context);
209     }
210 
211     // create a result object for the system id.
212     URL url = UrlFactory.getInstance().newUrl(systemId);
213 
214     // create an output stream for this url.
215     OutputStream out = url.openConnection().getOutputStream();
216 
217     // create a stream result for the output stream.
218     StreamResult streamResult = new StreamResult();
219     streamResult.setSystemId(systemId);
220     streamResult.setOutputStream(out);
221 
222     return streamResult;
223   }
224 
225   /**
226    * <p>Returns the Result object for the path attribute.</p>
227    * @param context the JXPathContext to evaluate against.
228    * @return a stream result for the path specified.
229    */
230   public Result createResultForPath(JXPathContext context)
231     throws Exception
232   {
233     // set the system id.
234     String path = getPath(context);
235 
236     // get the file object for the path.
237     File file = new File(path);
238 
239     // create the directories leading up to the path.
240     File parentFile = file.getParentFile();
241     if( !parentFile.exists() ) {
242       parentFile.mkdirs();
243     }
244 
245     // make sure that the file also exists.
246     file.createNewFile();
247 
248     // create an output stream for this url.
249     OutputStream out = new FileOutputStream(file);
250 
251     // create a stream result for the output stream.
252     StreamResult streamResult = new StreamResult();
253     streamResult.setSystemId(file.toURL().toExternalForm());
254     streamResult.setOutputStream(out);
255 
256     return streamResult;
257   }
258 
259   /**
260    * <p>Builds the result object for this element and sets it on the current sax pipeline configuration.</p>
261    */
262   public boolean execute(JXPathContext context)
263     throws Exception
264   {
265     Result result = null;
266 
267     if( hasSelect() ) {
268       result = createResultForSelect(context);
269     } else if( hasSystemId() || hasSystemIdDepricated() ) {
270       result = createResultForSystemId(context);
271     } else if( hasPath() ) {
272       result = createResultForPath(context);
273     } else {
274       throw new IllegalStateException("The system-id, path or select attribute must be set for the result tag.");
275     }
276 
277     // set the result in the pipeline config's composite stage.
278     PipelineCommand.getPipelineConfig().getCompositeStage().setResult(result);
279 
280     return false;
281   }
282 }