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 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  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. /**
  28. * A factory class for {@link ResourceResolver}s.
  29. */
  30. public final class ResourceResolverFactory {
  31. private ResourceResolverFactory() {
  32. }
  33. /**
  34. * Returns the default resource resolver, this is most basic resolver which can be used when
  35. * no there are no I/O or file access restrictions.
  36. *
  37. * @return the default resource resolver
  38. */
  39. public static ResourceResolver createDefaultResourceResolver() {
  40. return DefaultResourceResolver.INSTANCE;
  41. }
  42. /**
  43. * A helper merthod that creates an internal resource resolver using the default resover:
  44. * {@link ResourceResolverFactory#createDefaultResourceResolver()}.
  45. *
  46. * @param baseURI the base URI from which to resolve URIs
  47. * @return the default internal resource resolver
  48. */
  49. public static InternalResourceResolver createDefaultInternalResourceResolver(URI baseURI) {
  50. return new InternalResourceResolver(baseURI, createDefaultResourceResolver());
  51. }
  52. /**
  53. * Creates an interal resource resolver given a base URI and a resource resolver.
  54. *
  55. * @param baseURI the base URI from which to resolve URIs
  56. * @param resolver the resource resolver
  57. * @return the internal resource resolver
  58. */
  59. public static InternalResourceResolver createInternalResourceResolver(URI baseURI,
  60. ResourceResolver resolver) {
  61. return new InternalResourceResolver(baseURI, resolver);
  62. }
  63. /**
  64. * Creates a temporary-resource-schema aware resource resolver. Temporary resource URIs are
  65. * created by {@link TempResourceURIGenerator}.
  66. *
  67. * @param tempResourceResolver the temporary-resource-schema resolver to use
  68. * @param defaultResourceResolver the default resource resolver to use
  69. * @return the ressource resolver
  70. */
  71. public static ResourceResolver createTempAwareResourceResolver(
  72. TempResourceResolver tempResourceResolver,
  73. ResourceResolver defaultResourceResolver) {
  74. return new TempAwareResourceResolver(tempResourceResolver, defaultResourceResolver);
  75. }
  76. /**
  77. * This creates the builder class for binding URI schemas to implementations of
  78. * {@link ResourceResolver}. This allows users to define their own URI schemas such that they
  79. * have finer control over the acquisition of resources.
  80. *
  81. * @param defaultResolver the default resource resolver that should be used in the event that
  82. * none of the other registered resolvers match the schema
  83. * @return the schema aware {@link ResourceResolver} builder
  84. */
  85. public static SchemaAwareResourceResolverBuilder createSchemaAwareResourceResolverBuilder(
  86. ResourceResolver defaultResolver) {
  87. return new SchemaAwareResourceResolverBuilderImpl(defaultResolver);
  88. }
  89. private static final class DefaultResourceResolver implements ResourceResolver {
  90. private static final ResourceResolver INSTANCE = new DefaultResourceResolver();
  91. private final TempAwareResourceResolver delegate;
  92. private DefaultResourceResolver() {
  93. delegate = new TempAwareResourceResolver(new DefaultTempResourceResolver(),
  94. new NormalResourceResolver());
  95. }
  96. /** {@inheritDoc} */
  97. public Resource getResource(URI uri) throws IOException {
  98. return delegate.getResource(uri);
  99. }
  100. /** {@inheritDoc} */
  101. public OutputStream getOutputStream(URI uri) throws IOException {
  102. return delegate.getOutputStream(uri);
  103. }
  104. }
  105. private static final class TempAwareResourceResolver implements ResourceResolver {
  106. private final TempResourceResolver tempResourceResolver;
  107. private final ResourceResolver defaultResourceResolver;
  108. public TempAwareResourceResolver(TempResourceResolver tempResourceHandler,
  109. ResourceResolver defaultResourceResolver) {
  110. this.tempResourceResolver = tempResourceHandler;
  111. this.defaultResourceResolver = defaultResourceResolver;
  112. }
  113. private static boolean isTempUri(URI uri) {
  114. return TempResourceURIGenerator.isTempUri(uri);
  115. }
  116. /** {@inheritDoc} */
  117. public Resource getResource(URI uri) throws IOException {
  118. if (isTempUri(uri)) {
  119. return tempResourceResolver.getResource(uri.getPath());
  120. } else {
  121. return defaultResourceResolver.getResource(uri);
  122. }
  123. }
  124. /** {@inheritDoc} */
  125. public OutputStream getOutputStream(URI uri) throws IOException {
  126. if (isTempUri(uri)) {
  127. return tempResourceResolver.getOutputStream(uri.getPath());
  128. } else {
  129. return defaultResourceResolver.getOutputStream(uri);
  130. }
  131. }
  132. }
  133. private static class DefaultTempResourceResolver implements TempResourceResolver {
  134. private static File getTempFile(String path) throws IOException {
  135. File file = new File(System.getProperty("java.io.tmpdir"), path);
  136. file.deleteOnExit();
  137. return file;
  138. }
  139. /** {@inheritDoc} */
  140. public Resource getResource(String id) throws IOException {
  141. return new Resource(getTempFile(id).toURI().toURL().openStream());
  142. }
  143. /** {@inheritDoc} */
  144. public OutputStream getOutputStream(String id) throws IOException {
  145. File file = getTempFile(id);
  146. if (file.createNewFile()) {
  147. return new FileOutputStream(file);
  148. } else {
  149. throw new IOException("Filed to create temporary file: " + id);
  150. }
  151. }
  152. }
  153. private static class NormalResourceResolver implements ResourceResolver {
  154. public Resource getResource(URI uri) throws IOException {
  155. return new Resource(uri.toURL().openStream());
  156. }
  157. public OutputStream getOutputStream(URI uri) throws IOException {
  158. return new FileOutputStream(new File(uri));
  159. }
  160. }
  161. private static final class SchemaAwareResourceResolver implements ResourceResolver {
  162. private final Map<String, ResourceResolver> schemaHandlingResourceResolvers;
  163. private final ResourceResolver defaultResolver;
  164. private SchemaAwareResourceResolver(
  165. Map<String, ResourceResolver> schemaHandlingResourceResolvers,
  166. ResourceResolver defaultResolver) {
  167. this.schemaHandlingResourceResolvers = schemaHandlingResourceResolvers;
  168. this.defaultResolver = defaultResolver;
  169. }
  170. private ResourceResolver getResourceResolverForSchema(URI uri) {
  171. String schema = uri.getScheme();
  172. if (schemaHandlingResourceResolvers.containsKey(schema)) {
  173. return schemaHandlingResourceResolvers.get(schema);
  174. } else {
  175. return defaultResolver;
  176. }
  177. }
  178. /** {@inheritDoc} */
  179. public Resource getResource(URI uri) throws IOException {
  180. return getResourceResolverForSchema(uri).getResource(uri);
  181. }
  182. /** {@inheritDoc} */
  183. public OutputStream getOutputStream(URI uri) throws IOException {
  184. return getResourceResolverForSchema(uri).getOutputStream(uri);
  185. }
  186. }
  187. /**
  188. * Implementations of this interface will be builders for {@link ResourceResolver}, they bind
  189. * URI schemas to their respective resolver. This gives users more control over the mechanisms
  190. * by which URIs are resolved.
  191. * <p>
  192. * Here is an example of how this could be used:
  193. * </p>
  194. * <p><code>
  195. * SchemaAwareResourceResolverBuilder builder
  196. * = ResourceResolverFactory.createSchemaAwareResourceResolverBuilder(defaultResolver);
  197. * builder.registerResourceResolverForSchema("test", testResolver);
  198. * builder.registerResourceResolverForSchema("anotherTest", test2Resolver);
  199. * ResourceResolver resolver = builder.build();
  200. * </code></p>
  201. * This will result in all URIs for the form "test:///..." will be resolved using the
  202. * <code>testResolver</code> object; URIs of the form "anotherTest:///..." will be resolved
  203. * using <code>test2Resolver</code>; all other URIs will be resolved from the defaultResolver.
  204. */
  205. public interface SchemaAwareResourceResolverBuilder {
  206. /**
  207. * Register a schema with its respective {@link ResourceResolver}. This resolver will be
  208. * used as the only resolver for the specified schema.
  209. *
  210. * @param schema the schema to be used with the given resolver
  211. * @param resourceResolver the resource resolver
  212. */
  213. void registerResourceResolverForSchema(String schema, ResourceResolver resourceResolver);
  214. /**
  215. * Builds a {@link ResourceResolver} that will delegate to the respective resource resolver
  216. * when a registered URI schema is given
  217. *
  218. * @return a resolver that delegates to the appropriate schema resolver
  219. */
  220. ResourceResolver build();
  221. }
  222. private static final class CompletedSchemaAwareResourceResolverBuilder
  223. implements SchemaAwareResourceResolverBuilder {
  224. private static final SchemaAwareResourceResolverBuilder INSTANCE
  225. = new CompletedSchemaAwareResourceResolverBuilder();
  226. /** {@inheritDoc} */
  227. public ResourceResolver build() {
  228. throw new IllegalStateException("Resource resolver already built");
  229. }
  230. /** {@inheritDoc} */
  231. public void registerResourceResolverForSchema(String schema,
  232. ResourceResolver resourceResolver) {
  233. throw new IllegalStateException("Resource resolver already built");
  234. }
  235. }
  236. private static final class ActiveSchemaAwareResourceResolverBuilder
  237. implements SchemaAwareResourceResolverBuilder {
  238. private final Map<String, ResourceResolver> schemaHandlingResourceResolvers
  239. = new HashMap<String, ResourceResolver>();
  240. private final ResourceResolver defaultResolver;
  241. private ActiveSchemaAwareResourceResolverBuilder(ResourceResolver defaultResolver) {
  242. this.defaultResolver = defaultResolver;
  243. }
  244. /** {@inheritDoc} */
  245. public void registerResourceResolverForSchema(String schema,
  246. ResourceResolver resourceResolver) {
  247. schemaHandlingResourceResolvers.put(schema, resourceResolver);
  248. }
  249. /** {@inheritDoc} */
  250. public ResourceResolver build() {
  251. return new SchemaAwareResourceResolver(
  252. Collections.unmodifiableMap(schemaHandlingResourceResolvers), defaultResolver);
  253. }
  254. }
  255. private static final class SchemaAwareResourceResolverBuilderImpl
  256. implements SchemaAwareResourceResolverBuilder {
  257. private SchemaAwareResourceResolverBuilder delegate;
  258. private SchemaAwareResourceResolverBuilderImpl(ResourceResolver defaultResolver) {
  259. this.delegate = new ActiveSchemaAwareResourceResolverBuilder(defaultResolver);
  260. }
  261. /** {@inheritDoc} */
  262. public void registerResourceResolverForSchema(String schema,
  263. ResourceResolver resourceResolver) {
  264. delegate.registerResourceResolverForSchema(schema, resourceResolver);
  265. }
  266. /** {@inheritDoc} */
  267. public ResourceResolver build() {
  268. ResourceResolver resourceResolver = delegate.build();
  269. delegate = CompletedSchemaAwareResourceResolverBuilder.INSTANCE;
  270. return resourceResolver;
  271. }
  272. }
  273. }