You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ResourceResolverFactory.java 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. /*
  2. * Licensed to the Apache Software Foundation (ASF) under one or more
  3. * contributor license agreements. See the NOTICE file distributed with
  4. * this work for additional information regarding copyright ownership.
  5. * The ASF licenses this file to You under the Apache License, Version 2.0
  6. * (the "License"); you may not use this file except in compliance with
  7. * the License. You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. /* $Id$ */
  18. package org.apache.fop.apps.io;
  19. import java.io.File;
  20. import java.io.FileOutputStream;
  21. import java.io.IOException;
  22. import java.io.OutputStream;
  23. import java.net.URI;
  24. import java.util.Collections;
  25. import java.util.HashMap;
  26. import java.util.Map;
  27. import org.apache.xmlgraphics.io.Resource;
  28. import org.apache.xmlgraphics.io.ResourceResolver;
  29. import org.apache.xmlgraphics.io.TempResourceResolver;
  30. import org.apache.xmlgraphics.io.TempResourceURIGenerator;
  31. /**
  32. * A factory class for {@link ResourceResolver}s.
  33. */
  34. public final class ResourceResolverFactory {
  35. private ResourceResolverFactory() {
  36. }
  37. /**
  38. * Returns the default resource resolver, this is most basic resolver which can be used when
  39. * no there are no I/O or file access restrictions.
  40. *
  41. * @return the default resource resolver
  42. */
  43. public static ResourceResolver createDefaultResourceResolver() {
  44. return DefaultResourceResolver.INSTANCE;
  45. }
  46. /**
  47. * A helper merthod that creates an internal resource resolver using the default resover:
  48. * {@link ResourceResolverFactory#createDefaultResourceResolver()}.
  49. *
  50. * @param baseURI the base URI from which to resolve URIs
  51. * @return the default internal resource resolver
  52. */
  53. public static InternalResourceResolver createDefaultInternalResourceResolver(URI baseURI) {
  54. return new InternalResourceResolver(baseURI, createDefaultResourceResolver());
  55. }
  56. /**
  57. * Creates an interal resource resolver given a base URI and a resource resolver.
  58. *
  59. * @param baseURI the base URI from which to resolve URIs
  60. * @param resolver the resource resolver
  61. * @return the internal resource resolver
  62. */
  63. public static InternalResourceResolver createInternalResourceResolver(URI baseURI,
  64. ResourceResolver resolver) {
  65. return new InternalResourceResolver(baseURI, resolver);
  66. }
  67. /**
  68. * Creates a temporary-resource-scheme aware resource resolver. Temporary resource URIs are
  69. * created by {@link TempResourceURIGenerator}.
  70. *
  71. * @param tempResourceResolver the temporary-resource-scheme resolver to use
  72. * @param defaultResourceResolver the default resource resolver to use
  73. * @return the ressource resolver
  74. */
  75. public static ResourceResolver createTempAwareResourceResolver(
  76. TempResourceResolver tempResourceResolver,
  77. ResourceResolver defaultResourceResolver) {
  78. return new TempAwareResourceResolver(tempResourceResolver, defaultResourceResolver);
  79. }
  80. /**
  81. * This creates the builder class for binding URI schemes to implementations of
  82. * {@link ResourceResolver}. This allows users to define their own URI schemes such that they
  83. * have finer control over the acquisition of resources.
  84. *
  85. * @param defaultResolver the default resource resolver that should be used in the event that
  86. * none of the other registered resolvers match the scheme
  87. * @return the scheme aware {@link ResourceResolver} builder
  88. */
  89. public static SchemeAwareResourceResolverBuilder createSchemeAwareResourceResolverBuilder(
  90. ResourceResolver defaultResolver) {
  91. return new SchemeAwareResourceResolverBuilderImpl(defaultResolver);
  92. }
  93. private static final class DefaultResourceResolver implements ResourceResolver {
  94. private static final ResourceResolver INSTANCE = new DefaultResourceResolver();
  95. private final TempAwareResourceResolver delegate;
  96. private DefaultResourceResolver() {
  97. delegate = new TempAwareResourceResolver(new DefaultTempResourceResolver(),
  98. new NormalResourceResolver());
  99. }
  100. /** {@inheritDoc} */
  101. public Resource getResource(URI uri) throws IOException {
  102. return delegate.getResource(uri);
  103. }
  104. /** {@inheritDoc} */
  105. public OutputStream getOutputStream(URI uri) throws IOException {
  106. return delegate.getOutputStream(uri);
  107. }
  108. }
  109. private static final class TempAwareResourceResolver implements ResourceResolver {
  110. private final TempResourceResolver tempResourceResolver;
  111. private final ResourceResolver defaultResourceResolver;
  112. public TempAwareResourceResolver(TempResourceResolver tempResourceHandler,
  113. ResourceResolver defaultResourceResolver) {
  114. this.tempResourceResolver = tempResourceHandler;
  115. this.defaultResourceResolver = defaultResourceResolver;
  116. }
  117. private static boolean isTempURI(URI uri) {
  118. return TempResourceURIGenerator.isTempURI(uri);
  119. }
  120. /** {@inheritDoc} */
  121. public Resource getResource(URI uri) throws IOException {
  122. if (isTempURI(uri)) {
  123. return tempResourceResolver.getResource(uri.getPath());
  124. } else {
  125. return defaultResourceResolver.getResource(uri);
  126. }
  127. }
  128. /** {@inheritDoc} */
  129. public OutputStream getOutputStream(URI uri) throws IOException {
  130. if (isTempURI(uri)) {
  131. return tempResourceResolver.getOutputStream(uri.getPath());
  132. } else {
  133. return defaultResourceResolver.getOutputStream(uri);
  134. }
  135. }
  136. }
  137. private static class DefaultTempResourceResolver implements TempResourceResolver {
  138. private static File getTempFile(String path) throws IOException {
  139. File file = new File(System.getProperty("java.io.tmpdir"), path);
  140. file.deleteOnExit();
  141. return file;
  142. }
  143. /** {@inheritDoc} */
  144. public Resource getResource(String id) throws IOException {
  145. return new Resource(getTempFile(id).toURI().toURL().openStream());
  146. }
  147. /** {@inheritDoc} */
  148. public OutputStream getOutputStream(String id) throws IOException {
  149. File file = getTempFile(id);
  150. if (file.createNewFile()) {
  151. return new FileOutputStream(file);
  152. } else {
  153. throw new IOException("Filed to create temporary file: " + id);
  154. }
  155. }
  156. }
  157. private static class NormalResourceResolver implements ResourceResolver {
  158. public Resource getResource(URI uri) throws IOException {
  159. return new Resource(uri.toURL().openStream());
  160. }
  161. public OutputStream getOutputStream(URI uri) throws IOException {
  162. return new FileOutputStream(new File(uri));
  163. }
  164. }
  165. private static final class SchemeAwareResourceResolver implements ResourceResolver {
  166. private final Map<String, ResourceResolver> schemeHandlingResourceResolvers;
  167. private final ResourceResolver defaultResolver;
  168. private SchemeAwareResourceResolver(
  169. Map<String, ResourceResolver> schemEHandlingResourceResolvers,
  170. ResourceResolver defaultResolver) {
  171. this.schemeHandlingResourceResolvers = schemEHandlingResourceResolvers;
  172. this.defaultResolver = defaultResolver;
  173. }
  174. private ResourceResolver getResourceResolverForScheme(URI uri) {
  175. String scheme = uri.getScheme();
  176. if (schemeHandlingResourceResolvers.containsKey(scheme)) {
  177. return schemeHandlingResourceResolvers.get(scheme);
  178. } else {
  179. return defaultResolver;
  180. }
  181. }
  182. /** {@inheritDoc} */
  183. public Resource getResource(URI uri) throws IOException {
  184. return getResourceResolverForScheme(uri).getResource(uri);
  185. }
  186. /** {@inheritDoc} */
  187. public OutputStream getOutputStream(URI uri) throws IOException {
  188. return getResourceResolverForScheme(uri).getOutputStream(uri);
  189. }
  190. }
  191. /**
  192. * Implementations of this interface will be builders for {@link ResourceResolver}, they bind
  193. * URI schemes to their respective resolver. This gives users more control over the mechanisms
  194. * by which URIs are resolved.
  195. * <p>
  196. * Here is an example of how this could be used:
  197. * </p>
  198. * <p><code>
  199. * SchemeAwareResourceResolverBuilder builder
  200. * = ResourceResolverFactory.createSchemeAwareResourceResolverBuilder(defaultResolver);
  201. * builder.registerResourceResolverForScheme("test", testResolver);
  202. * builder.registerResourceResolverForScheme("anotherTest", test2Resolver);
  203. * ResourceResolver resolver = builder.build();
  204. * </code></p>
  205. * This will result in all URIs for the form "test:///..." will be resolved using the
  206. * <code>testResolver</code> object; URIs of the form "anotherTest:///..." will be resolved
  207. * using <code>test2Resolver</code>; all other URIs will be resolved from the defaultResolver.
  208. */
  209. public interface SchemeAwareResourceResolverBuilder {
  210. /**
  211. * Register a scheme with its respective {@link ResourceResolver}. This resolver will be
  212. * used as the only resolver for the specified scheme.
  213. *
  214. * @param scheme the scheme to be used with the given resolver
  215. * @param resourceResolver the resource resolver
  216. */
  217. void registerResourceResolverForScheme(String scheme, ResourceResolver resourceResolver);
  218. /**
  219. * Builds a {@link ResourceResolver} that will delegate to the respective resource resolver
  220. * when a registered URI scheme is given
  221. *
  222. * @return a resolver that delegates to the appropriate scheme resolver
  223. */
  224. ResourceResolver build();
  225. }
  226. private static final class CompletedSchemeAwareResourceResolverBuilder
  227. implements SchemeAwareResourceResolverBuilder {
  228. private static final SchemeAwareResourceResolverBuilder INSTANCE
  229. = new CompletedSchemeAwareResourceResolverBuilder();
  230. /** {@inheritDoc} */
  231. public ResourceResolver build() {
  232. throw new IllegalStateException("Resource resolver already built");
  233. }
  234. /** {@inheritDoc} */
  235. public void registerResourceResolverForScheme(String scheme,
  236. ResourceResolver resourceResolver) {
  237. throw new IllegalStateException("Resource resolver already built");
  238. }
  239. }
  240. private static final class ActiveSchemeAwareResourceResolverBuilder
  241. implements SchemeAwareResourceResolverBuilder {
  242. private final Map<String, ResourceResolver> schemeHandlingResourceResolvers
  243. = new HashMap<String, ResourceResolver>();
  244. private final ResourceResolver defaultResolver;
  245. private ActiveSchemeAwareResourceResolverBuilder(ResourceResolver defaultResolver) {
  246. this.defaultResolver = defaultResolver;
  247. }
  248. /** {@inheritDoc} */
  249. public void registerResourceResolverForScheme(String scheme,
  250. ResourceResolver resourceResolver) {
  251. schemeHandlingResourceResolvers.put(scheme, resourceResolver);
  252. }
  253. /** {@inheritDoc} */
  254. public ResourceResolver build() {
  255. return new SchemeAwareResourceResolver(
  256. Collections.unmodifiableMap(schemeHandlingResourceResolvers), defaultResolver);
  257. }
  258. }
  259. private static final class SchemeAwareResourceResolverBuilderImpl
  260. implements SchemeAwareResourceResolverBuilder {
  261. private SchemeAwareResourceResolverBuilder delegate;
  262. private SchemeAwareResourceResolverBuilderImpl(ResourceResolver defaultResolver) {
  263. this.delegate = new ActiveSchemeAwareResourceResolverBuilder(defaultResolver);
  264. }
  265. /** {@inheritDoc} */
  266. public void registerResourceResolverForScheme(String scheme,
  267. ResourceResolver resourceResolver) {
  268. delegate.registerResourceResolverForScheme(scheme, resourceResolver);
  269. }
  270. /** {@inheritDoc} */
  271. public ResourceResolver build() {
  272. ResourceResolver resourceResolver = delegate.build();
  273. delegate = CompletedSchemeAwareResourceResolverBuilder.INSTANCE;
  274. return resourceResolver;
  275. }
  276. }
  277. }